@needle-tools/engine 3.28.6-beta → 3.28.7-beta.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CHANGELOG.md +2259 -2253
- package/LICENSE.md +10 -10
- package/README.md +52 -52
- package/dist/needle-engine.js +7 -10
- package/dist/needle-engine.light.js +7 -10
- package/dist/needle-engine.light.min.js +7 -10
- package/dist/needle-engine.light.umd.cjs +7 -10
- package/dist/needle-engine.min.js +7 -10
- package/dist/needle-engine.umd.cjs +7 -10
- package/lib/engine/api.d.ts +52 -52
- package/lib/engine/api.js +51 -51
- package/lib/engine/assets/index.d.ts +1 -1
- package/lib/engine/assets/index.js +4 -4
- package/lib/engine/codegen/register_types.d.ts +1 -1
- package/lib/engine/codegen/register_types.js +439 -439
- package/lib/engine/debug/debug.d.ts +12 -12
- package/lib/engine/debug/debug.js +26 -26
- package/lib/engine/debug/debug_console.d.ts +2 -2
- package/lib/engine/debug/debug_console.js +204 -204
- package/lib/engine/debug/debug_overlay.d.ts +10 -10
- package/lib/engine/debug/debug_overlay.js +277 -277
- package/lib/engine/debug/index.d.ts +1 -1
- package/lib/engine/debug/index.js +1 -1
- package/lib/engine/engine_addressables.d.ts +75 -75
- package/lib/engine/engine_addressables.js +441 -441
- package/lib/engine/engine_application.d.ts +19 -19
- package/lib/engine/engine_application.js +45 -45
- package/lib/engine/engine_assetdatabase.d.ts +25 -25
- package/lib/engine/engine_assetdatabase.js +341 -341
- package/lib/engine/engine_camera.d.ts +6 -6
- package/lib/engine/engine_camera.js +23 -23
- package/lib/engine/engine_components.d.ts +17 -17
- package/lib/engine/engine_components.js +273 -273
- package/lib/engine/engine_components_internal.d.ts +11 -11
- package/lib/engine/engine_components_internal.js +41 -41
- package/lib/engine/engine_constants.d.ts +5 -5
- package/lib/engine/engine_constants.js +32 -32
- package/lib/engine/engine_context.d.ts +269 -269
- package/lib/engine/engine_context.js +1242 -1242
- package/lib/engine/engine_context_registry.d.ts +50 -50
- package/lib/engine/engine_context_registry.js +89 -89
- package/lib/engine/engine_coroutine.d.ts +4 -4
- package/lib/engine/engine_coroutine.js +21 -21
- package/lib/engine/engine_create_objects.d.ts +13 -13
- package/lib/engine/engine_create_objects.js +33 -33
- package/lib/engine/engine_default_parameters.d.ts +2 -2
- package/lib/engine/engine_default_parameters.js +3 -3
- package/lib/engine/engine_editor-sync.d.ts +17 -17
- package/lib/engine/engine_editor-sync.js +7 -7
- package/lib/engine/engine_element.d.ts +55 -55
- package/lib/engine/engine_element.js +559 -559
- package/lib/engine/engine_element_attributes.d.ts +49 -49
- package/lib/engine/engine_element_attributes.js +1 -1
- package/lib/engine/engine_element_extras.d.ts +6 -6
- package/lib/engine/engine_element_extras.js +13 -13
- package/lib/engine/engine_element_loading.d.ts +40 -40
- package/lib/engine/engine_element_loading.js +312 -312
- package/lib/engine/engine_element_overlay.d.ts +19 -19
- package/lib/engine/engine_element_overlay.js +143 -143
- package/lib/engine/engine_fileloader.d.ts +3 -3
- package/lib/engine/engine_fileloader.js +7 -7
- package/lib/engine/engine_gameobject.d.ts +39 -39
- package/lib/engine/engine_gameobject.js +559 -559
- package/lib/engine/engine_generic_utils.d.ts +1 -1
- package/lib/engine/engine_generic_utils.js +13 -13
- package/lib/engine/engine_gizmos.d.ts +26 -26
- package/lib/engine/engine_gizmos.js +282 -282
- package/lib/engine/engine_gltf.d.ts +13 -13
- package/lib/engine/engine_gltf.js +15 -15
- package/lib/engine/engine_gltf_builtin_components.d.ts +7 -7
- package/lib/engine/engine_gltf_builtin_components.js +298 -298
- package/lib/engine/engine_hot_reload.d.ts +5 -5
- package/lib/engine/engine_hot_reload.js +182 -182
- package/lib/engine/engine_input.d.ts +129 -129
- package/lib/engine/engine_input.js +799 -799
- package/lib/engine/engine_input_utils.d.ts +2 -2
- package/lib/engine/engine_input_utils.js +22 -22
- package/lib/engine/engine_instancing.d.ts +16 -16
- package/lib/engine/engine_instancing.js +36 -36
- package/lib/engine/engine_license.d.ts +4 -4
- package/lib/engine/engine_license.js +398 -398
- package/lib/engine/engine_lifecycle_api.d.ts +14 -14
- package/lib/engine/engine_lifecycle_api.js +24 -24
- package/lib/engine/engine_lifecycle_functions_internal.d.ts +6 -6
- package/lib/engine/engine_lifecycle_functions_internal.js +28 -28
- package/lib/engine/engine_lightdata.d.ts +23 -23
- package/lib/engine/engine_lightdata.js +86 -86
- package/lib/engine/engine_loaders.d.ts +7 -7
- package/lib/engine/engine_loaders.js +69 -69
- package/lib/engine/engine_mainloop_utils.d.ts +13 -13
- package/lib/engine/engine_mainloop_utils.js +426 -426
- package/lib/engine/engine_math.d.ts +43 -43
- package/lib/engine/engine_math.js +147 -147
- package/lib/engine/engine_networking.d.ts +176 -176
- package/lib/engine/engine_networking.js +649 -649
- package/lib/engine/engine_networking_auto.d.ts +24 -24
- package/lib/engine/engine_networking_auto.js +324 -324
- package/lib/engine/engine_networking_files.d.ts +23 -23
- package/lib/engine/engine_networking_files.js +176 -176
- package/lib/engine/engine_networking_files_default_components.d.ts +3 -3
- package/lib/engine/engine_networking_files_default_components.js +39 -39
- package/lib/engine/engine_networking_instantiate.d.ts +39 -39
- package/lib/engine/engine_networking_instantiate.js +302 -302
- package/lib/engine/engine_networking_peer.d.ts +15 -15
- package/lib/engine/engine_networking_peer.js +132 -132
- package/lib/engine/engine_networking_streams.d.ts +90 -90
- package/lib/engine/engine_networking_streams.js +428 -428
- package/lib/engine/engine_networking_types.d.ts +14 -14
- package/lib/engine/engine_networking_types.js +7 -7
- package/lib/engine/engine_networking_utils.d.ts +2 -2
- package/lib/engine/engine_networking_utils.js +20 -20
- package/lib/engine/engine_patcher.d.ts +10 -10
- package/lib/engine/engine_patcher.js +142 -142
- package/lib/engine/engine_physics.d.ts +115 -115
- package/lib/engine/engine_physics.js +228 -228
- package/lib/engine/engine_physics.types.d.ts +37 -37
- package/lib/engine/engine_physics.types.js +33 -33
- package/lib/engine/engine_physics_rapier.d.ts +112 -112
- package/lib/engine/engine_physics_rapier.js +1266 -1266
- package/lib/engine/engine_playerview.d.ts +26 -26
- package/lib/engine/engine_playerview.js +64 -64
- package/lib/engine/engine_scenelighting.d.ts +74 -74
- package/lib/engine/engine_scenelighting.js +285 -285
- package/lib/engine/engine_scenetools.d.ts +35 -35
- package/lib/engine/engine_scenetools.js +212 -212
- package/lib/engine/engine_serialization.d.ts +4 -4
- package/lib/engine/engine_serialization.js +4 -4
- package/lib/engine/engine_serialization_builtin_serializer.d.ts +62 -62
- package/lib/engine/engine_serialization_builtin_serializer.js +369 -369
- package/lib/engine/engine_serialization_core.d.ts +84 -84
- package/lib/engine/engine_serialization_core.js +576 -576
- package/lib/engine/engine_serialization_decorator.d.ts +15 -15
- package/lib/engine/engine_serialization_decorator.js +54 -54
- package/lib/engine/engine_setup.d.ts +1 -1
- package/lib/engine/engine_setup.js +2 -2
- package/lib/engine/engine_shaders.d.ts +31 -31
- package/lib/engine/engine_shaders.js +229 -229
- package/lib/engine/engine_shims.d.ts +3 -3
- package/lib/engine/engine_shims.js +22 -22
- package/lib/engine/engine_texture.d.ts +20 -20
- package/lib/engine/engine_texture.js +57 -57
- package/lib/engine/engine_three_utils.d.ts +51 -51
- package/lib/engine/engine_three_utils.js +342 -342
- package/lib/engine/engine_time.d.ts +19 -19
- package/lib/engine/engine_time.js +47 -47
- package/lib/engine/engine_types.d.ts +358 -358
- package/lib/engine/engine_types.js +72 -72
- package/lib/engine/engine_typestore.d.ts +16 -16
- package/lib/engine/engine_typestore.js +35 -35
- package/lib/engine/engine_util_decorator.d.ts +12 -12
- package/lib/engine/engine_util_decorator.js +115 -115
- package/lib/engine/engine_utils.d.ts +104 -104
- package/lib/engine/engine_utils.js +518 -518
- package/lib/engine/engine_utils_screenshot.d.ts +10 -10
- package/lib/engine/engine_utils_screenshot.js +70 -70
- package/lib/engine/engine_web_api.d.ts +12 -12
- package/lib/engine/engine_web_api.js +112 -112
- package/lib/engine/extensions/EXT_texture_exr.d.ts +8 -8
- package/lib/engine/extensions/EXT_texture_exr.js +32 -32
- package/lib/engine/extensions/NEEDLE_animator_controller_model.d.ts +116 -116
- package/lib/engine/extensions/NEEDLE_animator_controller_model.js +91 -91
- package/lib/engine/extensions/NEEDLE_components.d.ts +33 -33
- package/lib/engine/extensions/NEEDLE_components.js +206 -206
- package/lib/engine/extensions/NEEDLE_gameobject_data.d.ts +10 -10
- package/lib/engine/extensions/NEEDLE_gameobject_data.js +57 -57
- package/lib/engine/extensions/NEEDLE_lighting_settings.d.ts +38 -38
- package/lib/engine/extensions/NEEDLE_lighting_settings.js +183 -183
- package/lib/engine/extensions/NEEDLE_lightmaps.d.ts +18 -18
- package/lib/engine/extensions/NEEDLE_lightmaps.js +108 -108
- package/lib/engine/extensions/NEEDLE_persistent_assets.d.ts +11 -11
- package/lib/engine/extensions/NEEDLE_persistent_assets.js +63 -63
- package/lib/engine/extensions/NEEDLE_progressive.d.ts +41 -41
- package/lib/engine/extensions/NEEDLE_progressive.js +366 -366
- package/lib/engine/extensions/NEEDLE_render_objects.d.ts +13 -13
- package/lib/engine/extensions/NEEDLE_render_objects.js +159 -159
- package/lib/engine/extensions/NEEDLE_techniques_webgl.d.ts +39 -39
- package/lib/engine/extensions/NEEDLE_techniques_webgl.js +544 -544
- package/lib/engine/extensions/extension_resolver.d.ts +4 -4
- package/lib/engine/extensions/extension_resolver.js +1 -1
- package/lib/engine/extensions/extension_utils.d.ts +2 -2
- package/lib/engine/extensions/extension_utils.js +140 -140
- package/lib/engine/extensions/extensions.d.ts +21 -21
- package/lib/engine/extensions/extensions.js +94 -94
- package/lib/engine/extensions/index.d.ts +5 -5
- package/lib/engine/extensions/index.js +5 -5
- package/lib/engine/extensions/usage_tracker.d.ts +13 -13
- package/lib/engine/extensions/usage_tracker.js +61 -61
- package/lib/engine/js-extensions/Camera.d.ts +1 -1
- package/lib/engine/js-extensions/Camera.js +36 -36
- package/lib/engine/js-extensions/Layers.d.ts +3 -3
- package/lib/engine/js-extensions/Layers.js +19 -19
- package/lib/engine/js-extensions/index.d.ts +2 -2
- package/lib/engine/js-extensions/index.js +2 -2
- package/lib/engine/shaders/shaderData.d.ts +55 -55
- package/lib/engine/shaders/shaderData.js +58 -58
- package/lib/engine/tests/test_utils.d.ts +2 -2
- package/lib/engine/tests/test_utils.js +53 -53
- package/lib/engine-components/AlignmentConstraint.d.ts +10 -10
- package/lib/engine-components/AlignmentConstraint.js +39 -39
- package/lib/engine-components/Animation.d.ts +53 -53
- package/lib/engine-components/Animation.js +333 -333
- package/lib/engine-components/AnimationCurve.d.ts +16 -16
- package/lib/engine-components/AnimationCurve.js +97 -97
- package/lib/engine-components/AnimationUtils.d.ts +8 -8
- package/lib/engine-components/AnimationUtils.js +110 -110
- package/lib/engine-components/Animator.d.ts +81 -81
- package/lib/engine-components/Animator.js +229 -229
- package/lib/engine-components/AnimatorController.d.ts +57 -57
- package/lib/engine-components/AnimatorController.js +887 -887
- package/lib/engine-components/AudioListener.d.ts +7 -7
- package/lib/engine-components/AudioListener.js +30 -30
- package/lib/engine-components/AudioSource.d.ts +61 -61
- package/lib/engine-components/AudioSource.js +422 -422
- package/lib/engine-components/AvatarLoader.d.ts +19 -19
- package/lib/engine-components/AvatarLoader.js +173 -173
- package/lib/engine-components/AxesHelper.d.ts +9 -9
- package/lib/engine-components/AxesHelper.js +44 -44
- package/lib/engine-components/BasicIKConstraint.d.ts +9 -9
- package/lib/engine-components/BasicIKConstraint.js +43 -43
- package/lib/engine-components/BoxHelperComponent.d.ts +16 -16
- package/lib/engine-components/BoxHelperComponent.js +89 -89
- package/lib/engine-components/Camera.d.ts +70 -70
- package/lib/engine-components/Camera.js +450 -450
- package/lib/engine-components/CameraUtils.d.ts +1 -1
- package/lib/engine-components/CameraUtils.js +77 -77
- package/lib/engine-components/CharacterController.d.ts +46 -46
- package/lib/engine-components/CharacterController.js +227 -227
- package/lib/engine-components/Collider.d.ts +46 -46
- package/lib/engine-components/Collider.js +153 -153
- package/lib/engine-components/Component.d.ts +228 -228
- package/lib/engine-components/Component.js +541 -541
- package/lib/engine-components/ContactShadows.d.ts +23 -23
- package/lib/engine-components/ContactShadows.js +233 -233
- package/lib/engine-components/DeleteBox.d.ts +9 -9
- package/lib/engine-components/DeleteBox.js +30 -30
- package/lib/engine-components/DeviceFlag.d.ts +12 -12
- package/lib/engine-components/DeviceFlag.js +43 -43
- package/lib/engine-components/DragControls.d.ts +51 -51
- package/lib/engine-components/DragControls.js +516 -516
- package/lib/engine-components/DropListener.d.ts +15 -15
- package/lib/engine-components/DropListener.js +120 -120
- package/lib/engine-components/Duplicatable.d.ts +16 -16
- package/lib/engine-components/Duplicatable.js +150 -150
- package/lib/engine-components/EventList.d.ts +28 -28
- package/lib/engine-components/EventList.js +105 -105
- package/lib/engine-components/EventTrigger.d.ts +12 -12
- package/lib/engine-components/EventTrigger.js +50 -50
- package/lib/engine-components/EventType.d.ts +19 -19
- package/lib/engine-components/EventType.js +71 -71
- package/lib/engine-components/FlyControls.d.ts +7 -7
- package/lib/engine-components/FlyControls.js +25 -25
- package/lib/engine-components/Fog.d.ts +20 -20
- package/lib/engine-components/Fog.js +60 -60
- package/lib/engine-components/Gizmos.d.ts +12 -12
- package/lib/engine-components/Gizmos.js +60 -60
- package/lib/engine-components/GridHelper.d.ts +12 -12
- package/lib/engine-components/GridHelper.js +47 -47
- package/lib/engine-components/GroundProjection.d.ts +21 -21
- package/lib/engine-components/GroundProjection.js +97 -97
- package/lib/engine-components/Interactable.d.ts +10 -10
- package/lib/engine-components/Interactable.js +11 -11
- package/lib/engine-components/Joints.d.ts +19 -19
- package/lib/engine-components/Joints.js +51 -51
- package/lib/engine-components/LODGroup.d.ts +30 -30
- package/lib/engine-components/LODGroup.js +145 -145
- package/lib/engine-components/Light.d.ts +75 -75
- package/lib/engine-components/Light.js +475 -475
- package/lib/engine-components/LookAtConstraint.d.ts +7 -7
- package/lib/engine-components/LookAtConstraint.js +17 -17
- package/lib/engine-components/NestedGltf.d.ts +11 -11
- package/lib/engine-components/NestedGltf.js +74 -74
- package/lib/engine-components/Networking.d.ts +11 -11
- package/lib/engine-components/Networking.js +70 -70
- package/lib/engine-components/OffsetConstraint.d.ts +14 -14
- package/lib/engine-components/OffsetConstraint.js +65 -65
- package/lib/engine-components/OrbitControls.d.ts +111 -111
- package/lib/engine-components/OrbitControls.js +646 -646
- package/lib/engine-components/ParticleSystem.d.ts +145 -145
- package/lib/engine-components/ParticleSystem.js +1077 -1077
- package/lib/engine-components/ParticleSystemModules.d.ts +489 -489
- package/lib/engine-components/ParticleSystemModules.js +1667 -1667
- package/lib/engine-components/ParticleSystemSubEmitter.d.ts +25 -25
- package/lib/engine-components/ParticleSystemSubEmitter.js +86 -86
- package/lib/engine-components/PlayerColor.d.ts +13 -13
- package/lib/engine-components/PlayerColor.js +83 -83
- package/lib/engine-components/ReflectionProbe.d.ts +22 -22
- package/lib/engine-components/ReflectionProbe.js +181 -181
- package/lib/engine-components/Renderer.d.ts +112 -112
- package/lib/engine-components/Renderer.js +1029 -1029
- package/lib/engine-components/RendererLightmap.d.ts +19 -19
- package/lib/engine-components/RendererLightmap.js +127 -127
- package/lib/engine-components/RigidBody.d.ts +120 -120
- package/lib/engine-components/RigidBody.js +452 -452
- package/lib/engine-components/SceneSwitcher.d.ts +72 -72
- package/lib/engine-components/SceneSwitcher.js +583 -583
- package/lib/engine-components/ScreenCapture.d.ts +64 -64
- package/lib/engine-components/ScreenCapture.js +405 -405
- package/lib/engine-components/ShadowCatcher.d.ts +18 -18
- package/lib/engine-components/ShadowCatcher.js +144 -144
- package/lib/engine-components/Skybox.d.ts +23 -23
- package/lib/engine-components/Skybox.js +287 -287
- package/lib/engine-components/SmoothFollow.d.ts +14 -14
- package/lib/engine-components/SmoothFollow.js +63 -63
- package/lib/engine-components/SpatialTrigger.d.ts +27 -27
- package/lib/engine-components/SpatialTrigger.js +144 -144
- package/lib/engine-components/SpectatorCamera.d.ts +45 -45
- package/lib/engine-components/SpectatorCamera.js +593 -593
- package/lib/engine-components/SpriteRenderer.d.ts +48 -48
- package/lib/engine-components/SpriteRenderer.js +257 -257
- package/lib/engine-components/SyncedCamera.d.ts +27 -27
- package/lib/engine-components/SyncedCamera.js +187 -187
- package/lib/engine-components/SyncedRoom.d.ts +24 -24
- package/lib/engine-components/SyncedRoom.js +162 -162
- package/lib/engine-components/SyncedTransform.d.ts +35 -35
- package/lib/engine-components/SyncedTransform.js +265 -265
- package/lib/engine-components/TestRunner.d.ts +13 -13
- package/lib/engine-components/TestRunner.js +99 -99
- package/lib/engine-components/TransformGizmo.d.ts +16 -16
- package/lib/engine-components/TransformGizmo.js +148 -148
- package/lib/engine-components/VideoPlayer.d.ts +86 -86
- package/lib/engine-components/VideoPlayer.js +792 -792
- package/lib/engine-components/Voip.d.ts +29 -29
- package/lib/engine-components/Voip.js +203 -203
- package/lib/engine-components/XRFlag.d.ts +33 -33
- package/lib/engine-components/XRFlag.js +128 -128
- package/lib/engine-components/api.d.ts +15 -15
- package/lib/engine-components/api.js +15 -15
- package/lib/engine-components/avatar/AvatarBlink_Simple.d.ts +10 -10
- package/lib/engine-components/avatar/AvatarBlink_Simple.js +75 -75
- package/lib/engine-components/avatar/AvatarEyeLook_Rotation.d.ts +13 -13
- package/lib/engine-components/avatar/AvatarEyeLook_Rotation.js +74 -74
- package/lib/engine-components/avatar/Avatar_Brain_LookAt.d.ts +27 -27
- package/lib/engine-components/avatar/Avatar_Brain_LookAt.js +119 -119
- package/lib/engine-components/avatar/Avatar_MouthShapes.d.ts +13 -13
- package/lib/engine-components/avatar/Avatar_MouthShapes.js +78 -78
- package/lib/engine-components/avatar/Avatar_MustacheShake.d.ts +8 -8
- package/lib/engine-components/avatar/Avatar_MustacheShake.js +28 -28
- package/lib/engine-components/codegen/components.d.ts +216 -216
- package/lib/engine-components/codegen/components.js +217 -217
- package/lib/engine-components/debug/LogStats.d.ts +5 -5
- package/lib/engine-components/debug/LogStats.js +18 -18
- package/lib/engine-components/export/gltf/GltfExport.d.ts +25 -25
- package/lib/engine-components/export/gltf/GltfExport.js +215 -215
- package/lib/engine-components/export/index.d.ts +1 -1
- package/lib/engine-components/export/index.js +1 -1
- package/lib/engine-components/export/usdz/Extension.d.ts +10 -10
- package/lib/engine-components/export/usdz/Extension.js +1 -1
- package/lib/engine-components/export/usdz/ThreeUSDZExporter.d.ts +114 -114
- package/lib/engine-components/export/usdz/ThreeUSDZExporter.js +1211 -1211
- package/lib/engine-components/export/usdz/USDZExporter.d.ts +59 -59
- package/lib/engine-components/export/usdz/USDZExporter.js +450 -450
- package/lib/engine-components/export/usdz/extensions/Animation.d.ts +69 -69
- package/lib/engine-components/export/usdz/extensions/Animation.js +650 -650
- package/lib/engine-components/export/usdz/extensions/DocumentExtension.d.ts +5 -5
- package/lib/engine-components/export/usdz/extensions/DocumentExtension.js +6 -6
- package/lib/engine-components/export/usdz/extensions/USDZText.d.ts +55 -55
- package/lib/engine-components/export/usdz/extensions/USDZText.js +246 -246
- package/lib/engine-components/export/usdz/extensions/USDZUI.d.ts +8 -8
- package/lib/engine-components/export/usdz/extensions/USDZUI.js +100 -100
- package/lib/engine-components/export/usdz/extensions/behavior/Actions.d.ts +30 -30
- package/lib/engine-components/export/usdz/extensions/behavior/Actions.js +88 -88
- package/lib/engine-components/export/usdz/extensions/behavior/AudioExtension.d.ts +9 -9
- package/lib/engine-components/export/usdz/extensions/behavior/AudioExtension.js +52 -52
- package/lib/engine-components/export/usdz/extensions/behavior/Behaviour.d.ts +22 -22
- package/lib/engine-components/export/usdz/extensions/behavior/Behaviour.js +134 -134
- package/lib/engine-components/export/usdz/extensions/behavior/BehaviourComponents.d.ts +126 -126
- package/lib/engine-components/export/usdz/extensions/behavior/BehaviourComponents.js +824 -824
- package/lib/engine-components/export/usdz/extensions/behavior/BehavioursBuilder.d.ts +133 -133
- package/lib/engine-components/export/usdz/extensions/behavior/BehavioursBuilder.js +464 -464
- package/lib/engine-components/export/usdz/index.d.ts +3 -3
- package/lib/engine-components/export/usdz/index.js +2 -2
- package/lib/engine-components/export/usdz/utils/animationutils.d.ts +3 -3
- package/lib/engine-components/export/usdz/utils/animationutils.js +85 -85
- package/lib/engine-components/export/usdz/utils/quicklook.d.ts +2 -2
- package/lib/engine-components/export/usdz/utils/quicklook.js +35 -35
- package/lib/engine-components/export/usdz/utils/timeutils.d.ts +1 -1
- package/lib/engine-components/export/usdz/utils/timeutils.js +14 -14
- package/lib/engine-components/js-extensions/ExtensionUtils.d.ts +6 -6
- package/lib/engine-components/js-extensions/ExtensionUtils.js +65 -65
- package/lib/engine-components/js-extensions/Object3D.d.ts +2 -2
- package/lib/engine-components/js-extensions/Object3D.js +140 -140
- package/lib/engine-components/js-extensions/RGBAColor.d.ts +14 -14
- package/lib/engine-components/js-extensions/RGBAColor.js +49 -49
- package/lib/engine-components/js-extensions/index.d.ts +3 -3
- package/lib/engine-components/js-extensions/index.js +3 -3
- package/lib/engine-components/postprocessing/Effects/Antialiasing.d.ts +13 -13
- package/lib/engine-components/postprocessing/Effects/Antialiasing.js +45 -45
- package/lib/engine-components/postprocessing/Effects/Bloom.d.ts +12 -12
- package/lib/engine-components/postprocessing/Effects/Bloom.js +77 -77
- package/lib/engine-components/postprocessing/Effects/ChromaticAberration.d.ts +8 -8
- package/lib/engine-components/postprocessing/Effects/ChromaticAberration.js +38 -38
- package/lib/engine-components/postprocessing/Effects/ColorAdjustments.d.ts +12 -12
- package/lib/engine-components/postprocessing/Effects/ColorAdjustments.js +81 -81
- package/lib/engine-components/postprocessing/Effects/DepthOfField.d.ts +21 -21
- package/lib/engine-components/postprocessing/Effects/DepthOfField.js +97 -97
- package/lib/engine-components/postprocessing/Effects/Pixelation.d.ts +7 -7
- package/lib/engine-components/postprocessing/Effects/Pixelation.js +28 -28
- package/lib/engine-components/postprocessing/Effects/ScreenspaceAmbientOcclusion.d.ts +13 -13
- package/lib/engine-components/postprocessing/Effects/ScreenspaceAmbientOcclusion.js +86 -86
- package/lib/engine-components/postprocessing/Effects/ScreenspaceAmbientOcclusionN8.d.ts +24 -24
- package/lib/engine-components/postprocessing/Effects/ScreenspaceAmbientOcclusionN8.js +94 -94
- package/lib/engine-components/postprocessing/Effects/TiltShiftEffect.d.ts +13 -13
- package/lib/engine-components/postprocessing/Effects/TiltShiftEffect.js +62 -62
- package/lib/engine-components/postprocessing/Effects/Tonemapping.d.ts +16 -16
- package/lib/engine-components/postprocessing/Effects/Tonemapping.js +51 -51
- package/lib/engine-components/postprocessing/Effects/Vignette.d.ts +11 -11
- package/lib/engine-components/postprocessing/Effects/Vignette.js +56 -56
- package/lib/engine-components/postprocessing/PostProcessingEffect.d.ts +33 -33
- package/lib/engine-components/postprocessing/PostProcessingEffect.js +126 -126
- package/lib/engine-components/postprocessing/PostProcessingHandler.d.ts +22 -22
- package/lib/engine-components/postprocessing/PostProcessingHandler.js +201 -201
- package/lib/engine-components/postprocessing/Volume.d.ts +25 -25
- package/lib/engine-components/postprocessing/Volume.js +193 -193
- package/lib/engine-components/postprocessing/VolumeParameter.d.ts +22 -22
- package/lib/engine-components/postprocessing/VolumeParameter.js +80 -80
- package/lib/engine-components/postprocessing/VolumeProfile.d.ts +7 -7
- package/lib/engine-components/postprocessing/VolumeProfile.js +41 -41
- package/lib/engine-components/postprocessing/index.d.ts +4 -4
- package/lib/engine-components/postprocessing/index.js +4 -4
- package/lib/engine-components/timeline/PlayableDirector.d.ts +107 -107
- package/lib/engine-components/timeline/PlayableDirector.js +624 -624
- package/lib/engine-components/timeline/SignalAsset.d.ts +18 -18
- package/lib/engine-components/timeline/SignalAsset.js +124 -124
- package/lib/engine-components/timeline/TimelineModels.d.ts +88 -88
- package/lib/engine-components/timeline/TimelineModels.js +22 -22
- package/lib/engine-components/timeline/TimelineTracks.d.ts +90 -90
- package/lib/engine-components/timeline/TimelineTracks.js +825 -825
- package/lib/engine-components/timeline/index.d.ts +4 -4
- package/lib/engine-components/timeline/index.js +3 -3
- package/lib/engine-components/ui/BaseUIComponent.d.ts +31 -31
- package/lib/engine-components/ui/BaseUIComponent.js +161 -161
- package/lib/engine-components/ui/Button.d.ts +56 -56
- package/lib/engine-components/ui/Button.js +282 -282
- package/lib/engine-components/ui/Canvas.d.ts +67 -67
- package/lib/engine-components/ui/Canvas.js +382 -382
- package/lib/engine-components/ui/CanvasGroup.d.ts +15 -15
- package/lib/engine-components/ui/CanvasGroup.js +53 -53
- package/lib/engine-components/ui/EventSystem.d.ts +102 -102
- package/lib/engine-components/ui/EventSystem.js +641 -641
- package/lib/engine-components/ui/Graphic.d.ts +45 -45
- package/lib/engine-components/ui/Graphic.js +236 -236
- package/lib/engine-components/ui/Image.d.ts +27 -27
- package/lib/engine-components/ui/Image.js +107 -107
- package/lib/engine-components/ui/InputField.d.ts +34 -34
- package/lib/engine-components/ui/InputField.js +234 -234
- package/lib/engine-components/ui/Interfaces.d.ts +38 -38
- package/lib/engine-components/ui/Interfaces.js +12 -12
- package/lib/engine-components/ui/Layout.d.ts +72 -72
- package/lib/engine-components/ui/Layout.js +318 -318
- package/lib/engine-components/ui/Outline.d.ts +7 -7
- package/lib/engine-components/ui/Outline.js +20 -20
- package/lib/engine-components/ui/PointerEvents.d.ts +64 -64
- package/lib/engine-components/ui/PointerEvents.js +68 -68
- package/lib/engine-components/ui/RaycastUtils.d.ts +11 -11
- package/lib/engine-components/ui/RaycastUtils.js +67 -67
- package/lib/engine-components/ui/Raycaster.d.ts +18 -18
- package/lib/engine-components/ui/Raycaster.js +69 -69
- package/lib/engine-components/ui/RectTransform.d.ts +61 -61
- package/lib/engine-components/ui/RectTransform.js +343 -343
- package/lib/engine-components/ui/SpatialHtml.d.ts +6 -6
- package/lib/engine-components/ui/SpatialHtml.js +57 -57
- package/lib/engine-components/ui/Text.d.ts +74 -74
- package/lib/engine-components/ui/Text.js +534 -534
- package/lib/engine-components/ui/Utils.d.ts +23 -23
- package/lib/engine-components/ui/Utils.js +90 -90
- package/lib/engine-components/ui/index.d.ts +1 -1
- package/lib/engine-components/ui/index.js +1 -1
- package/lib/engine-components/utils/LookAt.d.ts +13 -13
- package/lib/engine-components/utils/LookAt.js +59 -59
- package/lib/engine-components/utils/OpenURL.d.ts +21 -21
- package/lib/engine-components/utils/OpenURL.js +124 -124
- package/lib/engine-components/webxr/WebARCameraBackground.d.ts +19 -19
- package/lib/engine-components/webxr/WebARCameraBackground.js +193 -193
- package/lib/engine-components/webxr/WebARSessionRoot.d.ts +38 -38
- package/lib/engine-components/webxr/WebARSessionRoot.js +407 -407
- package/lib/engine-components/webxr/WebXR.d.ts +110 -110
- package/lib/engine-components/webxr/WebXR.js +672 -672
- package/lib/engine-components/webxr/WebXRAvatar.d.ts +61 -61
- package/lib/engine-components/webxr/WebXRAvatar.js +289 -289
- package/lib/engine-components/webxr/WebXRController.d.ts +154 -154
- package/lib/engine-components/webxr/WebXRController.js +1028 -1028
- package/lib/engine-components/webxr/WebXRGrabRendering.d.ts +42 -42
- package/lib/engine-components/webxr/WebXRGrabRendering.js +137 -137
- package/lib/engine-components/webxr/WebXRImageTracking.d.ts +49 -49
- package/lib/engine-components/webxr/WebXRImageTracking.js +336 -336
- package/lib/engine-components/webxr/WebXRPlaneTracking.d.ts +49 -49
- package/lib/engine-components/webxr/WebXRPlaneTracking.js +372 -372
- package/lib/engine-components/webxr/WebXRRig.d.ts +4 -4
- package/lib/engine-components/webxr/WebXRRig.js +19 -19
- package/lib/engine-components/webxr/WebXRSync.d.ts +54 -54
- package/lib/engine-components/webxr/WebXRSync.js +410 -410
- package/lib/engine-components/webxr/index.d.ts +4 -4
- package/lib/engine-components/webxr/index.js +4 -4
- package/lib/engine-components-experimental/Presentation.d.ts +6 -6
- package/lib/engine-components-experimental/Presentation.js +9 -9
- package/lib/engine-components-experimental/api.d.ts +1 -1
- package/lib/engine-components-experimental/api.js +1 -1
- package/lib/engine-components-experimental/networking/PlayerSync.d.ts +50 -50
- package/lib/engine-components-experimental/networking/PlayerSync.js +200 -200
- package/lib/engine-schemes/api.d.ts +1 -1
- package/lib/engine-schemes/api.js +1 -1
- package/lib/engine-schemes/schemes.d.ts +7 -7
- package/lib/engine-schemes/schemes.js +19 -19
- package/lib/engine-schemes/synced-camera-model.d.ts +25 -25
- package/lib/engine-schemes/synced-camera-model.js +67 -67
- package/lib/engine-schemes/synced-transform-model.d.ts +31 -31
- package/lib/engine-schemes/synced-transform-model.js +66 -66
- package/lib/engine-schemes/transform.d.ts +12 -12
- package/lib/engine-schemes/transform.js +39 -39
- package/lib/engine-schemes/vec3.d.ts +11 -11
- package/lib/engine-schemes/vec3.js +29 -29
- package/lib/engine-schemes/vec4.d.ts +12 -12
- package/lib/engine-schemes/vec4.js +33 -33
- package/lib/engine-schemes/vr-user-state-buffer.d.ts +36 -36
- package/lib/engine-schemes/vr-user-state-buffer.js +103 -103
- package/lib/include/three/ARButton.d.ts +3 -3
- package/lib/include/three/ARButton.js +151 -151
- package/lib/include/three/EXT_mesh_gpu_instancing_exporter.d.ts +6 -6
- package/lib/include/three/EXT_mesh_gpu_instancing_exporter.js +45 -45
- package/lib/include/three/VRButton.d.ts +5 -5
- package/lib/include/three/VRButton.js +118 -118
- package/lib/needle-engine.d.ts +6 -6
- package/lib/needle-engine.js +49 -49
- package/package.json +1 -1
- package/plugins/common/config.cjs +14 -14
- package/plugins/common/config.js +19 -19
- package/plugins/common/generator.js +10 -10
- package/plugins/common/license.cjs +30 -30
- package/plugins/common/version.js +11 -11
- package/plugins/next/license.cjs +4 -4
- package/plugins/next/next.js +70 -70
- package/plugins/types/index.d.ts +1 -1
- package/plugins/types/needleConfig.d.ts +21 -21
- package/plugins/types/userconfig.d.ts +42 -42
- package/plugins/vite/alias.js +70 -70
- package/plugins/vite/build.js +19 -19
- package/plugins/vite/config.js +73 -73
- package/plugins/vite/copyfiles.js +134 -134
- package/plugins/vite/defines.js +45 -45
- package/plugins/vite/dependency-watcher.js +224 -224
- package/plugins/vite/drop-client.js +76 -76
- package/plugins/vite/drop.js +82 -82
- package/plugins/vite/editor-connection.js +121 -121
- package/plugins/vite/facebook-instant-games.js +99 -99
- package/plugins/vite/gzip.js +5 -5
- package/plugins/vite/imports-logger.js +143 -143
- package/plugins/vite/index.js +81 -81
- package/plugins/vite/license.js +42 -42
- package/plugins/vite/meta.js +149 -149
- package/plugins/vite/peer.js +31 -31
- package/plugins/vite/poster-client.js +59 -59
- package/plugins/vite/poster.js +73 -73
- package/plugins/vite/reload-client.js +15 -15
- package/plugins/vite/reload.js +363 -363
- package/plugins/vite/transform-codegen.js +55 -55
- package/plugins/vite/vite-4.4-hack.js +31 -31
- package/src/engine/api.ts +54 -54
- package/src/engine/assets/index.ts +4 -4
- package/src/engine/codegen/register_types.ts +441 -441
- package/src/engine/debug/debug.ts +29 -29
- package/src/engine/debug/debug_console.ts +213 -213
- package/src/engine/debug/debug_overlay.ts +283 -283
- package/src/engine/engine.ts +13 -13
- package/src/engine/engine_addressables.ts +494 -494
- package/src/engine/engine_application.ts +53 -53
- package/src/engine/engine_assetdatabase.ts +383 -383
- package/src/engine/engine_camera.ts +32 -32
- package/src/engine/engine_components.ts +266 -266
- package/src/engine/engine_components_internal.ts +42 -42
- package/src/engine/engine_constants.ts +42 -42
- package/src/engine/engine_context.ts +1386 -1386
- package/src/engine/engine_context_registry.ts +103 -103
- package/src/engine/engine_coroutine.ts +24 -24
- package/src/engine/engine_create_objects.ts +39 -39
- package/src/engine/engine_default_parameters.ts +3 -3
- package/src/engine/engine_editor-sync.ts +29 -29
- package/src/engine/engine_element.ts +592 -592
- package/src/engine/engine_element_attributes.ts +61 -61
- package/src/engine/engine_element_extras.ts +16 -16
- package/src/engine/engine_element_loading.ts +341 -341
- package/src/engine/engine_element_overlay.ts +160 -160
- package/src/engine/engine_fileloader.js +8 -8
- package/src/engine/engine_gameobject.ts +621 -621
- package/src/engine/engine_generic_utils.js +13 -13
- package/src/engine/engine_gizmos.ts +321 -321
- package/src/engine/engine_gltf.ts +30 -30
- package/src/engine/engine_gltf_builtin_components.ts +350 -350
- package/src/engine/engine_hot_reload.ts +196 -196
- package/src/engine/engine_input.ts +879 -879
- package/src/engine/engine_input_utils.ts +23 -23
- package/src/engine/engine_instancing.ts +42 -42
- package/src/engine/engine_license.ts +413 -413
- package/src/engine/engine_lifecycle_api.ts +29 -29
- package/src/engine/engine_lifecycle_functions_internal.ts +36 -36
- package/src/engine/engine_lightdata.ts +113 -113
- package/src/engine/engine_loaders.ts +77 -77
- package/src/engine/engine_mainloop_utils.ts +431 -431
- package/src/engine/engine_math.ts +174 -174
- package/src/engine/engine_networking.ts +742 -742
- package/src/engine/engine_networking_auto.ts +373 -373
- package/src/engine/engine_networking_files.ts +206 -206
- package/src/engine/engine_networking_files_default_components.ts +54 -54
- package/src/engine/engine_networking_instantiate.ts +362 -362
- package/src/engine/engine_networking_peer.ts +158 -158
- package/src/engine/engine_networking_streams.ts +489 -489
- package/src/engine/engine_networking_types.ts +18 -18
- package/src/engine/engine_networking_utils.ts +23 -23
- package/src/engine/engine_networking_websocket.ts +2 -2
- package/src/engine/engine_patcher.ts +199 -199
- package/src/engine/engine_physics.ts +287 -287
- package/src/engine/engine_physics.types.ts +43 -43
- package/src/engine/engine_physics_rapier.ts +1385 -1385
- package/src/engine/engine_playerview.ts +79 -79
- package/src/engine/engine_scenelighting.ts +313 -313
- package/src/engine/engine_scenetools.ts +242 -242
- package/src/engine/engine_serialization.ts +6 -6
- package/src/engine/engine_serialization_builtin_serializer.ts +415 -415
- package/src/engine/engine_serialization_core.ts +680 -680
- package/src/engine/engine_serialization_decorator.ts +68 -68
- package/src/engine/engine_setup.ts +1 -1
- package/src/engine/engine_shaders.ts +242 -242
- package/src/engine/engine_shims.ts +28 -28
- package/src/engine/engine_texture.ts +70 -70
- package/src/engine/engine_three_utils.ts +382 -382
- package/src/engine/engine_time.ts +55 -55
- package/src/engine/engine_types.ts +489 -489
- package/src/engine/engine_typestore.ts +41 -41
- package/src/engine/engine_util_decorator.ts +134 -134
- package/src/engine/engine_utils.ts +605 -605
- package/src/engine/engine_utils_screenshot.ts +84 -84
- package/src/engine/engine_web_api.ts +119 -119
- package/src/engine/extensions/EXT_texture_exr.ts +49 -49
- package/src/engine/extensions/NEEDLE_animator_controller_model.ts +193 -193
- package/src/engine/extensions/NEEDLE_components.ts +250 -250
- package/src/engine/extensions/NEEDLE_gameobject_data.ts +82 -82
- package/src/engine/extensions/NEEDLE_lighting_settings.ts +210 -210
- package/src/engine/extensions/NEEDLE_lightmaps.ts +130 -130
- package/src/engine/extensions/NEEDLE_persistent_assets.ts +75 -75
- package/src/engine/extensions/NEEDLE_progressive.ts +412 -412
- package/src/engine/extensions/NEEDLE_render_objects.ts +209 -209
- package/src/engine/extensions/NEEDLE_techniques_webgl.ts +618 -618
- package/src/engine/extensions/extension_resolver.ts +4 -4
- package/src/engine/extensions/extension_utils.ts +149 -149
- package/src/engine/extensions/extensions.ts +118 -118
- package/src/engine/extensions/index.ts +4 -4
- package/src/engine/extensions/usage_tracker.ts +95 -95
- package/src/engine/js-extensions/Camera.ts +34 -34
- package/src/engine/js-extensions/Layers.ts +19 -19
- package/src/engine/js-extensions/index.ts +1 -1
- package/src/engine/shaders/shaderData.ts +67 -67
- package/src/engine/tests/test_utils.ts +63 -63
- package/src/engine-components/AlignmentConstraint.ts +35 -35
- package/src/engine-components/Animation.ts +345 -345
- package/src/engine-components/AnimationCurve.ts +83 -83
- package/src/engine-components/AnimationUtils.ts +117 -117
- package/src/engine-components/Animator.ts +243 -243
- package/src/engine-components/AnimatorController.ts +1020 -1020
- package/src/engine-components/AudioListener.ts +32 -32
- package/src/engine-components/AudioSource.ts +419 -419
- package/src/engine-components/AvatarLoader.ts +204 -204
- package/src/engine-components/AxesHelper.ts +33 -33
- package/src/engine-components/BasicIKConstraint.ts +53 -53
- package/src/engine-components/BoxCollider.ts +1 -1
- package/src/engine-components/BoxHelperComponent.ts +100 -100
- package/src/engine-components/Camera.ts +454 -454
- package/src/engine-components/CameraUtils.ts +89 -89
- package/src/engine-components/CharacterController.ts +243 -243
- package/src/engine-components/Collider.ts +160 -160
- package/src/engine-components/Component.ts +670 -670
- package/src/engine-components/ContactShadows.ts +265 -265
- package/src/engine-components/DeleteBox.ts +35 -35
- package/src/engine-components/DeviceFlag.ts +42 -42
- package/src/engine-components/DragControls.ts +574 -574
- package/src/engine-components/DropListener.ts +112 -112
- package/src/engine-components/Duplicatable.ts +146 -146
- package/src/engine-components/EventList.ts +125 -125
- package/src/engine-components/EventTrigger.ts +47 -47
- package/src/engine-components/EventType.ts +87 -87
- package/src/engine-components/FlyControls.ts +31 -31
- package/src/engine-components/Fog.ts +59 -59
- package/src/engine-components/Gizmos.ts +52 -52
- package/src/engine-components/GridHelper.ts +40 -40
- package/src/engine-components/GroundProjection.ts +97 -97
- package/src/engine-components/Interactable.ts +18 -18
- package/src/engine-components/Joints.ts +51 -51
- package/src/engine-components/LODGroup.ts +145 -145
- package/src/engine-components/Light.ts +493 -493
- package/src/engine-components/LookAtConstraint.ts +11 -11
- package/src/engine-components/NestedGltf.ts +70 -70
- package/src/engine-components/Networking.ts +72 -72
- package/src/engine-components/OffsetConstraint.ts +59 -59
- package/src/engine-components/OrbitControls.ts +653 -653
- package/src/engine-components/ParticleSystem.ts +1192 -1192
- package/src/engine-components/ParticleSystemModules.ts +1481 -1481
- package/src/engine-components/ParticleSystemSubEmitter.ts +110 -110
- package/src/engine-components/PlayerColor.ts +93 -93
- package/src/engine-components/ReflectionProbe.ts +192 -192
- package/src/engine-components/Renderer.ts +1125 -1125
- package/src/engine-components/RendererLightmap.ts +145 -145
- package/src/engine-components/RigidBody.ts +453 -453
- package/src/engine-components/SceneSwitcher.ts +594 -594
- package/src/engine-components/ScreenCapture.ts +437 -437
- package/src/engine-components/ShadowCatcher.ts +149 -149
- package/src/engine-components/Skybox.ts +281 -281
- package/src/engine-components/SmoothFollow.ts +57 -57
- package/src/engine-components/SpatialTrigger.ts +142 -142
- package/src/engine-components/SpectatorCamera.ts +675 -675
- package/src/engine-components/SphereCollider.ts +1 -1
- package/src/engine-components/SpriteRenderer.ts +244 -244
- package/src/engine-components/SyncedCamera.ts +208 -208
- package/src/engine-components/SyncedRoom.ts +166 -166
- package/src/engine-components/SyncedTransform.ts +336 -336
- package/src/engine-components/TestRunner.ts +114 -114
- package/src/engine-components/TransformGizmo.ts +157 -157
- package/src/engine-components/VideoPlayer.ts +831 -831
- package/src/engine-components/Voip.ts +214 -214
- package/src/engine-components/XRFlag.ts +138 -138
- package/src/engine-components/api.ts +22 -22
- package/src/engine-components/avatar/AvatarBlink_Simple.ts +67 -67
- package/src/engine-components/avatar/AvatarEyeLook_Rotation.ts +68 -68
- package/src/engine-components/avatar/Avatar_Brain_LookAt.ts +136 -136
- package/src/engine-components/avatar/Avatar_MouthShapes.ts +81 -81
- package/src/engine-components/avatar/Avatar_MustacheShake.ts +28 -28
- package/src/engine-components/codegen/components.ts +216 -216
- package/src/engine-components/debug/LogStats.ts +21 -21
- package/src/engine-components/export/gltf/GltfExport.ts +231 -231
- package/src/engine-components/export/usdz/Extension.ts +11 -11
- package/src/engine-components/export/usdz/ThreeUSDZExporter.ts +1773 -1773
- package/src/engine-components/export/usdz/USDZExporter.ts +477 -477
- package/src/engine-components/export/usdz/extensions/Animation.ts +774 -774
- package/src/engine-components/export/usdz/extensions/DocumentExtension.ts +9 -9
- package/src/engine-components/export/usdz/extensions/USDZText.ts +287 -287
- package/src/engine-components/export/usdz/extensions/USDZUI.ts +119 -119
- package/src/engine-components/export/usdz/extensions/behavior/Actions.ts +98 -98
- package/src/engine-components/export/usdz/extensions/behavior/AudioExtension.ts +67 -67
- package/src/engine-components/export/usdz/extensions/behavior/Behaviour.ts +202 -202
- package/src/engine-components/export/usdz/extensions/behavior/BehaviourComponents.ts +963 -963
- package/src/engine-components/export/usdz/extensions/behavior/BehavioursBuilder.ts +517 -517
- package/src/engine-components/export/usdz/index.ts +2 -2
- package/src/engine-components/export/usdz/utils/animationutils.ts +100 -100
- package/src/engine-components/export/usdz/utils/quicklook.ts +42 -42
- package/src/engine-components/export/usdz/utils/timeutils.ts +19 -19
- package/src/engine-components/js-extensions/ExtensionUtils.ts +81 -81
- package/src/engine-components/js-extensions/Object3D.ts +181 -181
- package/src/engine-components/js-extensions/RGBAColor.ts +54 -54
- package/src/engine-components/js-extensions/Vector.ts +16 -16
- package/src/engine-components/js-extensions/index.ts +2 -2
- package/src/engine-components/postprocessing/Effects/Antialiasing.ts +51 -51
- package/src/engine-components/postprocessing/Effects/Bloom.ts +76 -76
- package/src/engine-components/postprocessing/Effects/ChromaticAberration.ts +35 -35
- package/src/engine-components/postprocessing/Effects/ColorAdjustments.ts +96 -96
- package/src/engine-components/postprocessing/Effects/DepthOfField.ts +93 -93
- package/src/engine-components/postprocessing/Effects/Pixelation.ts +26 -26
- package/src/engine-components/postprocessing/Effects/ScreenspaceAmbientOcclusion.ts +84 -84
- package/src/engine-components/postprocessing/Effects/ScreenspaceAmbientOcclusionN8.ts +98 -98
- package/src/engine-components/postprocessing/Effects/TiltShiftEffect.ts +55 -55
- package/src/engine-components/postprocessing/Effects/Tonemapping.ts +54 -54
- package/src/engine-components/postprocessing/Effects/Vignette.ts +54 -54
- package/src/engine-components/postprocessing/PostProcessingEffect.ts +148 -148
- package/src/engine-components/postprocessing/PostProcessingHandler.ts +232 -232
- package/src/engine-components/postprocessing/Volume.ts +216 -216
- package/src/engine-components/postprocessing/VolumeParameter.ts +92 -92
- package/src/engine-components/postprocessing/VolumeProfile.ts +40 -40
- package/src/engine-components/postprocessing/index.ts +3 -3
- package/src/engine-components/timeline/PlayableDirector.ts +666 -666
- package/src/engine-components/timeline/SignalAsset.ts +138 -138
- package/src/engine-components/timeline/TimelineModels.ts +93 -93
- package/src/engine-components/timeline/TimelineTracks.ts +906 -906
- package/src/engine-components/timeline/index.ts +3 -3
- package/src/engine-components/ui/BaseUIComponent.ts +195 -195
- package/src/engine-components/ui/Button.ts +283 -283
- package/src/engine-components/ui/Canvas.ts +390 -390
- package/src/engine-components/ui/CanvasGroup.ts +49 -49
- package/src/engine-components/ui/EventSystem.ts +736 -736
- package/src/engine-components/ui/Graphic.ts +255 -255
- package/src/engine-components/ui/Image.ts +102 -102
- package/src/engine-components/ui/InputField.ts +290 -290
- package/src/engine-components/ui/Interfaces.ts +57 -57
- package/src/engine-components/ui/Layout.ts +322 -322
- package/src/engine-components/ui/Outline.ts +12 -12
- package/src/engine-components/ui/PointerEvents.ts +118 -118
- package/src/engine-components/ui/RaycastUtils.ts +68 -68
- package/src/engine-components/ui/Raycaster.ts +73 -73
- package/src/engine-components/ui/RectTransform.ts +364 -364
- package/src/engine-components/ui/SpatialHtml.ts +63 -63
- package/src/engine-components/ui/Text.ts +572 -572
- package/src/engine-components/ui/Utils.ts +110 -110
- package/src/engine-components/utils/LookAt.ts +65 -65
- package/src/engine-components/utils/OpenURL.ts +118 -118
- package/src/engine-components/webxr/WebARCameraBackground.ts +224 -224
- package/src/engine-components/webxr/WebARSessionRoot.ts +446 -446
- package/src/engine-components/webxr/WebXR.ts +761 -761
- package/src/engine-components/webxr/WebXRAvatar.ts +356 -356
- package/src/engine-components/webxr/WebXRController.ts +1168 -1168
- package/src/engine-components/webxr/WebXRGrabRendering.ts +150 -150
- package/src/engine-components/webxr/WebXRImageTracking.ts +371 -371
- package/src/engine-components/webxr/WebXRPlaneTracking.ts +429 -429
- package/src/engine-components/webxr/WebXRRig.ts +21 -21
- package/src/engine-components/webxr/WebXRSync.ts +463 -463
- package/src/engine-components/webxr/index.ts +3 -3
- package/src/engine-components-experimental/Presentation.ts +12 -12
- package/src/engine-components-experimental/networking/PlayerSync.ts +217 -217
- package/src/engine-schemes/COMPILE_SCHEMES.bat +3 -3
- package/src/engine-schemes/COMPILE_TS.bat +11 -11
- package/src/engine-schemes/schemes.ts +27 -27
- package/src/engine-schemes/synced-camera-model.ts +92 -92
- package/src/engine-schemes/synced-transform-model.ts +90 -90
- package/src/engine-schemes/syncedCamera.fbs +10 -10
- package/src/engine-schemes/transform.ts +50 -50
- package/src/engine-schemes/transforms.fbs +25 -25
- package/src/engine-schemes/vec.fbs +19 -19
- package/src/engine-schemes/vec2.ts +33 -33
- package/src/engine-schemes/vec3.ts +38 -38
- package/src/engine-schemes/vec4.ts +43 -43
- package/src/engine-schemes/vr-user-state-buffer.ts +138 -138
- package/src/engine-schemes/vrUserStateBuffer.fbs +16 -16
- package/src/include/draco/draco_decoder.js +34 -34
- package/src/include/draco/draco_wasm_wrapper.js +117 -117
- package/src/include/ktx2/basis_transcoder.js +21 -21
- package/src/include/needle/arial-msdf.json +1471 -1471
- package/src/include/three/ARButton.js +231 -231
- package/src/include/three/DragControls.js +231 -231
- package/src/include/three/EXT_mesh_gpu_instancing_exporter.js +66 -66
- package/src/include/three/VRButton.js +194 -194
- package/src/needle-engine.ts +55 -55
- package/src/engine/dist/api.js +0 -73
- package/src/engine/dist/api.js.meta +0 -7
- package/src/engine/dist/engine_networking_streams.js +0 -474
- package/src/engine/dist/engine_networking_streams.js.meta +0 -7
- package/src/engine-schemes/dist/api.js +0 -17
- package/src/engine-schemes/dist/api.js.meta +0 -7
- package/src/engine-schemes/dist/schemes.js +0 -25
- package/src/engine-schemes/dist/schemes.js.meta +0 -7
- package/src/engine-schemes/dist/synced-camera-model.js +0 -74
- package/src/engine-schemes/dist/synced-camera-model.js.meta +0 -7
- package/src/engine-schemes/dist/synced-transform-model.js +0 -73
- package/src/engine-schemes/dist/synced-transform-model.js.meta +0 -7
- package/src/engine-schemes/dist/transform.js +0 -46
- package/src/engine-schemes/dist/transform.js.meta +0 -7
- package/src/engine-schemes/dist/vec2.js +0 -32
- package/src/engine-schemes/dist/vec2.js.meta +0 -7
- package/src/engine-schemes/dist/vec3.js +0 -36
- package/src/engine-schemes/dist/vec3.js.meta +0 -7
- package/src/engine-schemes/dist/vec4.js +0 -40
- package/src/engine-schemes/dist/vec4.js.meta +0 -7
- package/src/engine-schemes/dist/vr-user-state-buffer.js +0 -110
- package/src/engine-schemes/dist/vr-user-state-buffer.js.meta +0 -7
- package/src/engine-schemes/flatc.exe +0 -0
|
@@ -1,826 +1,826 @@
|
|
|
1
|
-
import { PlayableDirector } from "./PlayableDirector.js";
|
|
2
|
-
import * as Models from "./TimelineModels.js";
|
|
3
|
-
import { GameObject } from "../Component.js";
|
|
4
|
-
import { SignalReceiver } from "./SignalAsset.js";
|
|
5
|
-
import { Audio, AudioLoader, Euler, Quaternion, QuaternionKeyframeTrack, Vector3, VectorKeyframeTrack } from "three";
|
|
6
|
-
import { getParam, resolveUrl } from "../../engine/engine_utils.js";
|
|
7
|
-
import { AudioSource } from "../AudioSource.js";
|
|
8
|
-
import { Animator } from "../Animator.js";
|
|
9
|
-
import { setObjectAnimated } from "../AnimationUtils.js";
|
|
10
|
-
import { isDevEnvironment } from "../../engine/debug/index.js";
|
|
11
|
-
const debug = getParam("debugtimeline");
|
|
12
|
-
export class TrackHandler {
|
|
13
|
-
director;
|
|
14
|
-
track;
|
|
15
|
-
get muted() { return this.track.muted; }
|
|
16
|
-
set muted(val) {
|
|
17
|
-
if (val !== this.track.muted) {
|
|
18
|
-
this.track.muted = val;
|
|
19
|
-
this.onMuteChanged?.call(this);
|
|
20
|
-
}
|
|
21
|
-
}
|
|
22
|
-
*forEachClip(backwards = false) {
|
|
23
|
-
if (!this.track?.clips)
|
|
24
|
-
return;
|
|
25
|
-
if (backwards) {
|
|
26
|
-
for (let i = this.track.clips.length - 1; i >= 0; i--) {
|
|
27
|
-
yield this.track.clips[i];
|
|
28
|
-
}
|
|
29
|
-
}
|
|
30
|
-
else {
|
|
31
|
-
for (const clip of this.track.clips) {
|
|
32
|
-
yield clip;
|
|
33
|
-
}
|
|
34
|
-
}
|
|
35
|
-
}
|
|
36
|
-
getClipTime(time, model) {
|
|
37
|
-
return model.clipIn + (time - model.start) * model.timeScale;
|
|
38
|
-
}
|
|
39
|
-
getClipTimeNormalized(time, model) {
|
|
40
|
-
return (time - model.start) / model.duration;
|
|
41
|
-
}
|
|
42
|
-
evaluateWeight(time, index, models, isActive = true) {
|
|
43
|
-
if (index < 0 || index >= models.length)
|
|
44
|
-
return 0;
|
|
45
|
-
const model = models[index];
|
|
46
|
-
if (isActive || time >= model.start && time <= model.end) {
|
|
47
|
-
let weight = 1;
|
|
48
|
-
const isBlendingWithNext = false;
|
|
49
|
-
// this blending with next clips is already baked into easeIn/easeOut
|
|
50
|
-
// if (allowBlendWithNext && index + 1 < models.length) {
|
|
51
|
-
// const next = models[index + 1];
|
|
52
|
-
// const nextWeight = (time - next.start) / (model.end - next.start);
|
|
53
|
-
// isBlendingWithNext = nextWeight > 0;
|
|
54
|
-
// weight = 1 - nextWeight;
|
|
55
|
-
// }
|
|
56
|
-
if (model.easeInDuration > 0) {
|
|
57
|
-
const easeIn = Math.min((time - model.start) / model.easeInDuration, 1);
|
|
58
|
-
weight *= easeIn;
|
|
59
|
-
}
|
|
60
|
-
if (model.easeOutDuration > 0 && !isBlendingWithNext) {
|
|
61
|
-
const easeOut = Math.min((model.end - time) / model.easeOutDuration, 1);
|
|
62
|
-
weight *= easeOut;
|
|
63
|
-
}
|
|
64
|
-
return weight;
|
|
65
|
-
}
|
|
66
|
-
return 0;
|
|
67
|
-
}
|
|
68
|
-
}
|
|
69
|
-
class AnimationClipOffsetData {
|
|
70
|
-
clip;
|
|
71
|
-
rootPositionOffset;
|
|
72
|
-
rootQuaternionOffset;
|
|
73
|
-
get hasOffsets() { return this.rootPositionOffset !== undefined || this.rootQuaternionOffset !== undefined; }
|
|
74
|
-
// not necessary
|
|
75
|
-
rootStartPosition;
|
|
76
|
-
rootEndPosition;
|
|
77
|
-
rootStartQuaternion;
|
|
78
|
-
rootEndQuaternion;
|
|
79
|
-
constructor(action) {
|
|
80
|
-
const clip = action.getClip();
|
|
81
|
-
this.clip = clip;
|
|
82
|
-
const root = action.getRoot();
|
|
83
|
-
const rootPositionTrackName = root.name + ".position";
|
|
84
|
-
const rootRotationTrackName = root.name + ".quaternion";
|
|
85
|
-
if (debug)
|
|
86
|
-
console.log(clip.name, clip.tracks, rootPositionTrackName);
|
|
87
|
-
for (const track of clip.tracks) {
|
|
88
|
-
if (track.times.length <= 0)
|
|
89
|
-
continue;
|
|
90
|
-
if (track.name.endsWith(rootPositionTrackName)) {
|
|
91
|
-
this.rootStartPosition = new Vector3().fromArray(track.values, 0);
|
|
92
|
-
this.rootEndPosition = new Vector3().fromArray(track.values, track.values.length - 3);
|
|
93
|
-
this.rootPositionOffset = this.rootEndPosition.clone().sub(this.rootStartPosition);
|
|
94
|
-
if (debug)
|
|
95
|
-
console.log(this.rootPositionOffset);
|
|
96
|
-
// this.rootPositionOffset.set(0, 0, 0);
|
|
97
|
-
}
|
|
98
|
-
else if (track.name.endsWith(rootRotationTrackName)) {
|
|
99
|
-
this.rootStartQuaternion = new Quaternion().fromArray(track.values, 0);
|
|
100
|
-
this.rootEndQuaternion = new Quaternion().fromArray(track.values, track.values.length - 4);
|
|
101
|
-
this.rootQuaternionOffset = this.rootEndQuaternion.clone().multiply(this.rootStartQuaternion);
|
|
102
|
-
if (debug) {
|
|
103
|
-
const euler = new Euler().setFromQuaternion(this.rootQuaternionOffset);
|
|
104
|
-
console.log("ROT", euler);
|
|
105
|
-
}
|
|
106
|
-
}
|
|
107
|
-
}
|
|
108
|
-
}
|
|
109
|
-
}
|
|
110
|
-
// TODO: add support for clip clamp modes (loop, pingpong, clamp)
|
|
111
|
-
export class AnimationTrackHandler extends TrackHandler {
|
|
112
|
-
models = [];
|
|
113
|
-
trackOffset;
|
|
114
|
-
target;
|
|
115
|
-
/** The AnimationMixer, should be shared with the animator if an animator is bound */
|
|
116
|
-
mixer;
|
|
117
|
-
clips = [];
|
|
118
|
-
actions = [];
|
|
119
|
-
/** holds data/info about clips differences */
|
|
120
|
-
_actionOffsets = [];
|
|
121
|
-
_didBind = false;
|
|
122
|
-
_animator = null;
|
|
123
|
-
// Using this callback instead of onEnable etc
|
|
124
|
-
// because we want to re-enable the animator when the director is at the end and wrap mode is set to none
|
|
125
|
-
// in which case the director is stopped (but not disabled)
|
|
126
|
-
// which means we want to notify the object that it's not animated anymore
|
|
127
|
-
// and the animator can then take over
|
|
128
|
-
onStateChanged() {
|
|
129
|
-
if (this._animator)
|
|
130
|
-
setObjectAnimated(this._animator.gameObject, this, this.director.isPlaying);
|
|
131
|
-
}
|
|
132
|
-
createHooks(clipModel, clip) {
|
|
133
|
-
if (clip.tracks?.length <= 0) {
|
|
134
|
-
console.warn("No tracks in AnimationClip", clip);
|
|
135
|
-
return;
|
|
136
|
-
}
|
|
137
|
-
// we only want to hook into the binding of the root object
|
|
138
|
-
// TODO: test with a clip with multiple roots
|
|
139
|
-
const parts = clip.tracks[0].name.split(".");
|
|
140
|
-
const rootName = parts[parts.length - 2];
|
|
141
|
-
const positionTrackName = rootName + ".position";
|
|
142
|
-
const rotationTrackName = rootName + ".quaternion";
|
|
143
|
-
let foundPositionTrack = false;
|
|
144
|
-
let foundRotationTrack = false;
|
|
145
|
-
for (const t of clip.tracks) {
|
|
146
|
-
if (t.name.endsWith(positionTrackName)) {
|
|
147
|
-
foundPositionTrack = true;
|
|
148
|
-
this.createPositionInterpolant(clip, clipModel, t);
|
|
149
|
-
}
|
|
150
|
-
else if (t.name.endsWith(rotationTrackName)) {
|
|
151
|
-
foundRotationTrack = true;
|
|
152
|
-
this.createRotationInterpolant(clip, clipModel, t);
|
|
153
|
-
}
|
|
154
|
-
}
|
|
155
|
-
// ensure we always have a position and rotation track so we can apply offsets in interpolator
|
|
156
|
-
// TODO: this currently assumes that there is only one root always that has offsets so it only does create the interpolator for the first track which might be incorrect. In general it would probably be better if we would not create additional tracks but apply the offsets for these objects elsewhere!?
|
|
157
|
-
if (!foundPositionTrack || !foundRotationTrack) {
|
|
158
|
-
const root = this.mixer?.getRoot();
|
|
159
|
-
const track = clip.tracks[0];
|
|
160
|
-
const indexOfProperty = track.name.lastIndexOf(".");
|
|
161
|
-
const baseName = track.name.substring(0, indexOfProperty);
|
|
162
|
-
const objName = baseName.substring(baseName.lastIndexOf(".") + 1);
|
|
163
|
-
const targetObj = root.getObjectByName(objName);
|
|
164
|
-
// TODO can't animate unnamed objects which use GUID as name this way, need scene.getObjectByProperty('uuid', objectName);
|
|
165
|
-
// This should be right but needs testing:
|
|
166
|
-
// const parsedPath = PropertyBinding.parseTrackName(track.name);
|
|
167
|
-
// const targetObj = PropertyBinding.findNode(root, parsedPath.nodeName);
|
|
168
|
-
if (targetObj) {
|
|
169
|
-
if (!foundPositionTrack) {
|
|
170
|
-
const trackName = baseName + ".position";
|
|
171
|
-
if (debug)
|
|
172
|
-
console.warn("Create position track", objName, targetObj);
|
|
173
|
-
// apply initial local position so it doesnt get flipped or otherwise changed
|
|
174
|
-
const pos = targetObj.position;
|
|
175
|
-
const track = new VectorKeyframeTrack(trackName, [0, clip.duration], [pos.x, pos.y, pos.z, pos.x, pos.y, pos.z]);
|
|
176
|
-
clip.tracks.push(track);
|
|
177
|
-
this.createPositionInterpolant(clip, clipModel, track);
|
|
178
|
-
}
|
|
179
|
-
else if (!foundRotationTrack) {
|
|
180
|
-
const trackName = clip.tracks[0].name.substring(0, indexOfProperty) + ".quaternion";
|
|
181
|
-
if (debug)
|
|
182
|
-
console.warn("Create quaternion track", objName, targetObj);
|
|
183
|
-
const rot = targetObj.quaternion;
|
|
184
|
-
const track = new QuaternionKeyframeTrack(trackName, [0, clip.duration], [rot.x, rot.y, rot.z, rot.w, rot.x, rot.y, rot.z, rot.w]);
|
|
185
|
-
clip.tracks.push(track);
|
|
186
|
-
this.createRotationInterpolant(clip, clipModel, track);
|
|
187
|
-
}
|
|
188
|
-
}
|
|
189
|
-
}
|
|
190
|
-
}
|
|
191
|
-
bind() {
|
|
192
|
-
if (this._didBind)
|
|
193
|
-
return;
|
|
194
|
-
this._didBind = true;
|
|
195
|
-
if (debug)
|
|
196
|
-
console.log(this.models);
|
|
197
|
-
// the object being animated
|
|
198
|
-
if (this.mixer)
|
|
199
|
-
this.target = this.mixer.getRoot();
|
|
200
|
-
else
|
|
201
|
-
console.warn("No mixer was assigned to animation track");
|
|
202
|
-
for (const action of this.actions) {
|
|
203
|
-
const off = new AnimationClipOffsetData(action);
|
|
204
|
-
this._actionOffsets.push(off);
|
|
205
|
-
}
|
|
206
|
-
if (this.target) {
|
|
207
|
-
// We need to disable the animator component in case it also animates
|
|
208
|
-
// which overrides the timeline
|
|
209
|
-
this._animator = GameObject.getComponent(this.target, Animator) ?? null;
|
|
210
|
-
if (this._animator) {
|
|
211
|
-
setObjectAnimated(this._animator.gameObject, this, true);
|
|
212
|
-
}
|
|
213
|
-
}
|
|
214
|
-
// Clip Offsets
|
|
215
|
-
for (const model of this.models) {
|
|
216
|
-
const clipData = model.asset;
|
|
217
|
-
const pos = clipData.position;
|
|
218
|
-
const rot = clipData.rotation;
|
|
219
|
-
if (pos && pos.x !== undefined) {
|
|
220
|
-
if (!pos.isVector3) {
|
|
221
|
-
clipData.position = new Vector3(pos.x, pos.y, pos.z);
|
|
222
|
-
}
|
|
223
|
-
if (!rot.isQuaternion) {
|
|
224
|
-
clipData.rotation = new Quaternion(rot.x, rot.y, rot.z, rot.w);
|
|
225
|
-
}
|
|
226
|
-
}
|
|
227
|
-
}
|
|
228
|
-
this.ensureTrackOffsets();
|
|
229
|
-
}
|
|
230
|
-
ensureTrackOffsets() {
|
|
231
|
-
if (this.trackOffset) {
|
|
232
|
-
const pos = this.trackOffset.position;
|
|
233
|
-
if (pos) {
|
|
234
|
-
if (!pos.isVector3) {
|
|
235
|
-
this.trackOffset.position = new Vector3(pos.x, pos.y, pos.z);
|
|
236
|
-
}
|
|
237
|
-
}
|
|
238
|
-
const rot = this.trackOffset.rotation;
|
|
239
|
-
if (rot) {
|
|
240
|
-
if (!rot.isQuaternion) {
|
|
241
|
-
this.trackOffset.rotation = new Quaternion(rot.x, rot.y, rot.z, rot.w);
|
|
242
|
-
}
|
|
243
|
-
}
|
|
244
|
-
}
|
|
245
|
-
}
|
|
246
|
-
_useclipOffsets = true;
|
|
247
|
-
_totalOffsetPosition = new Vector3();
|
|
248
|
-
_totalOffsetRotation = new Quaternion();
|
|
249
|
-
_totalOffsetPosition2 = new Vector3();
|
|
250
|
-
_totalOffsetRotation2 = new Quaternion();
|
|
251
|
-
_summedPos = new Vector3();
|
|
252
|
-
_tempPos = new Vector3();
|
|
253
|
-
_summedRot = new Quaternion();
|
|
254
|
-
_tempRot = new Quaternion();
|
|
255
|
-
_clipRotQuat = new Quaternion();
|
|
256
|
-
evaluate(time) {
|
|
257
|
-
if (this.track.muted)
|
|
258
|
-
return;
|
|
259
|
-
if (!this.mixer)
|
|
260
|
-
return;
|
|
261
|
-
this.bind();
|
|
262
|
-
// if (this._animator && this.director.isPlaying && this.director.weight > 0) this._animator.enabled = false;
|
|
263
|
-
this._totalOffsetPosition.set(0, 0, 0);
|
|
264
|
-
this._totalOffsetRotation.set(0, 0, 0, 1);
|
|
265
|
-
this._totalOffsetPosition2.set(0, 0, 0);
|
|
266
|
-
this._totalOffsetRotation2.set(0, 0, 0, 1);
|
|
267
|
-
let activeClips = 0;
|
|
268
|
-
let blend = 0;
|
|
269
|
-
let didPostExtrapolate = false;
|
|
270
|
-
let didPreExtrapolate = false;
|
|
271
|
-
// The total weight is used to blend with the animator controller active states
|
|
272
|
-
let totalWeight = 0;
|
|
273
|
-
for (let i = 0; i < this.clips.length; i++) {
|
|
274
|
-
const model = this.models[i];
|
|
275
|
-
const action = this.actions[i];
|
|
276
|
-
const clipModel = model.asset;
|
|
277
|
-
action.weight = 0;
|
|
278
|
-
const isInTimeRange = time >= model.start && time <= model.end;
|
|
279
|
-
const preExtrapolation = model.preExtrapolationMode;
|
|
280
|
-
const postExtrapolation = model.postExtrapolationMode;
|
|
281
|
-
const nextClip = i < this.clips.length - 1 ? this.models[i + 1] : null;
|
|
282
|
-
let isActive = isInTimeRange;
|
|
283
|
-
let doPreExtrapolate = false;
|
|
284
|
-
if (!isActive && !didPostExtrapolate && model.end < time && postExtrapolation !== Models.ClipExtrapolation.None) {
|
|
285
|
-
// use post extrapolate if its the last clip of the next clip has not yet started
|
|
286
|
-
if (!nextClip || nextClip.start > time) {
|
|
287
|
-
isActive = true;
|
|
288
|
-
didPostExtrapolate = true;
|
|
289
|
-
}
|
|
290
|
-
}
|
|
291
|
-
else if (i == 0 && !isActive && !didPreExtrapolate && model.start > time && preExtrapolation !== Models.ClipExtrapolation.None) {
|
|
292
|
-
if (!nextClip || nextClip.start < time) {
|
|
293
|
-
isActive = true;
|
|
294
|
-
doPreExtrapolate = true;
|
|
295
|
-
didPreExtrapolate = true;
|
|
296
|
-
}
|
|
297
|
-
}
|
|
298
|
-
if (isActive) {
|
|
299
|
-
// const clip = this.clips[i];
|
|
300
|
-
let weight = 1;
|
|
301
|
-
weight *= this.evaluateWeight(time, i, this.models, isActive);
|
|
302
|
-
weight *= this.director.weight;
|
|
303
|
-
let handleLoop = isInTimeRange;
|
|
304
|
-
if (doPreExtrapolate) {
|
|
305
|
-
switch (preExtrapolation) {
|
|
306
|
-
case Models.ClipExtrapolation.Hold:
|
|
307
|
-
// Nothing to do
|
|
308
|
-
break;
|
|
309
|
-
case Models.ClipExtrapolation.Loop:
|
|
310
|
-
// TODO: this is not correct yet
|
|
311
|
-
time += model.start;
|
|
312
|
-
handleLoop = true;
|
|
313
|
-
break;
|
|
314
|
-
default:
|
|
315
|
-
time += model.start;
|
|
316
|
-
handleLoop = true;
|
|
317
|
-
break;
|
|
318
|
-
}
|
|
319
|
-
}
|
|
320
|
-
// TODO: handle clipIn again
|
|
321
|
-
let t = this.getClipTime(time, model);
|
|
322
|
-
let loops = 0;
|
|
323
|
-
const duration = clipModel.duration;
|
|
324
|
-
// This is the actual duration of the clip in the timeline (with clipping and scale)
|
|
325
|
-
// const clipDuration = (model.end - model.start) * model.timeScale;
|
|
326
|
-
if (doPreExtrapolate) {
|
|
327
|
-
if (preExtrapolation === Models.ClipExtrapolation.Hold) {
|
|
328
|
-
t = 0;
|
|
329
|
-
}
|
|
330
|
-
}
|
|
331
|
-
if (handleLoop) {
|
|
332
|
-
if (clipModel.loop) {
|
|
333
|
-
// const t0 = t - .001;
|
|
334
|
-
loops += Math.floor(t / (duration + .000001));
|
|
335
|
-
while (t > duration) {
|
|
336
|
-
t -= duration;
|
|
337
|
-
}
|
|
338
|
-
}
|
|
339
|
-
}
|
|
340
|
-
else if (!isInTimeRange) {
|
|
341
|
-
if (didPostExtrapolate) {
|
|
342
|
-
switch (postExtrapolation) {
|
|
343
|
-
case Models.ClipExtrapolation.Hold:
|
|
344
|
-
t = this.getClipTime(model.end, model);
|
|
345
|
-
break;
|
|
346
|
-
case Models.ClipExtrapolation.Loop:
|
|
347
|
-
t %= duration;
|
|
348
|
-
break;
|
|
349
|
-
case Models.ClipExtrapolation.PingPong:
|
|
350
|
-
const loops = Math.floor(t / duration);
|
|
351
|
-
const invert = loops % 2 !== 0;
|
|
352
|
-
t %= duration;
|
|
353
|
-
if (invert)
|
|
354
|
-
t = duration - t;
|
|
355
|
-
break;
|
|
356
|
-
}
|
|
357
|
-
}
|
|
358
|
-
}
|
|
359
|
-
if (model.reversed === true)
|
|
360
|
-
action.time = action.getClip().duration - t;
|
|
361
|
-
else
|
|
362
|
-
action.time = t;
|
|
363
|
-
action.timeScale = 0;
|
|
364
|
-
const effectiveWeight = Math.max(0, weight);
|
|
365
|
-
action.weight = effectiveWeight;
|
|
366
|
-
totalWeight += effectiveWeight;
|
|
367
|
-
action.clampWhenFinished = false;
|
|
368
|
-
if (!action.isRunning())
|
|
369
|
-
action.play();
|
|
370
|
-
// console.log(action.time, action.weight);
|
|
371
|
-
if (this._useclipOffsets) {
|
|
372
|
-
const totalPosition = activeClips == 0 ? this._totalOffsetPosition : this._totalOffsetPosition2;
|
|
373
|
-
const totalRotation = activeClips == 0 ? this._totalOffsetRotation : this._totalOffsetRotation2;
|
|
374
|
-
if (activeClips < 1)
|
|
375
|
-
blend = 1 - weight;
|
|
376
|
-
activeClips += 1;
|
|
377
|
-
const summedPos = this._summedPos.set(0, 0, 0);
|
|
378
|
-
const tempPos = this._tempPos.set(0, 0, 0);
|
|
379
|
-
const summedRot = this._summedRot.identity();
|
|
380
|
-
const tempRot = this._tempRot.identity();
|
|
381
|
-
const clipOffsetRot = clipModel.rotation;
|
|
382
|
-
if (clipOffsetRot) {
|
|
383
|
-
this._clipRotQuat.identity();
|
|
384
|
-
this._clipRotQuat.slerp(clipOffsetRot, weight);
|
|
385
|
-
}
|
|
386
|
-
const offsets = this._actionOffsets[i];
|
|
387
|
-
if (offsets.hasOffsets) {
|
|
388
|
-
for (let i = 0; i < loops; i++) {
|
|
389
|
-
if (offsets.rootPositionOffset)
|
|
390
|
-
tempPos.copy(offsets.rootPositionOffset);
|
|
391
|
-
else
|
|
392
|
-
tempPos.set(0, 0, 0);
|
|
393
|
-
tempPos.applyQuaternion(summedRot);
|
|
394
|
-
if (this._clipRotQuat)
|
|
395
|
-
tempPos.applyQuaternion(this._clipRotQuat);
|
|
396
|
-
if (offsets.rootQuaternionOffset) {
|
|
397
|
-
// console.log(new Euler().setFromQuaternion(offsets.rootQuaternionOffset).y.toFixed(2));
|
|
398
|
-
tempRot.copy(offsets.rootQuaternionOffset);
|
|
399
|
-
summedRot.multiply(tempRot);
|
|
400
|
-
}
|
|
401
|
-
summedPos.add(tempPos);
|
|
402
|
-
}
|
|
403
|
-
}
|
|
404
|
-
if (this._clipRotQuat)
|
|
405
|
-
totalRotation.multiply(this._clipRotQuat);
|
|
406
|
-
totalRotation.multiply(summedRot);
|
|
407
|
-
if (clipModel.position)
|
|
408
|
-
summedPos.add(clipModel.position);
|
|
409
|
-
totalPosition.add(summedPos);
|
|
410
|
-
}
|
|
411
|
-
}
|
|
412
|
-
}
|
|
413
|
-
if (this._useclipOffsets) {
|
|
414
|
-
this._totalOffsetPosition.lerp(this._totalOffsetPosition2, blend);
|
|
415
|
-
this._totalOffsetRotation.slerp(this._totalOffsetRotation2, blend);
|
|
416
|
-
}
|
|
417
|
-
if (this["__mixerError"] === undefined && (debug || isDevEnvironment()) && this._animator?.runtimeAnimatorController?.mixer && this.mixer !== this._animator?.runtimeAnimatorController?.mixer) {
|
|
418
|
-
this["__mixerError"] = true;
|
|
419
|
-
console.error("AnimationTrack mixer is not shared with the animator controller - this might result in the timeline to not animate properly. Please report a bug to the Needle Engine team!", this);
|
|
420
|
-
}
|
|
421
|
-
if (this._animator?.runtimeAnimatorController) {
|
|
422
|
-
// If the Timeline is running then the timeline track takes control over the animatorcontroller
|
|
423
|
-
// we calculate the weight left for the animatorcontroller actions
|
|
424
|
-
const weightLeft = Math.max(0, 1 - totalWeight);
|
|
425
|
-
this._animator?.runtimeAnimatorController?.update(weightLeft);
|
|
426
|
-
}
|
|
427
|
-
else {
|
|
428
|
-
this.mixer.update(time);
|
|
429
|
-
}
|
|
430
|
-
}
|
|
431
|
-
createRotationInterpolant(_clip, _clipModel, track) {
|
|
432
|
-
const createInterpolantOriginal = track.createInterpolant.bind(track);
|
|
433
|
-
const quat = new Quaternion();
|
|
434
|
-
this.ensureTrackOffsets();
|
|
435
|
-
const trackOffsetRot = this.trackOffset?.rotation;
|
|
436
|
-
track.createInterpolant = () => {
|
|
437
|
-
const createdInterpolant = createInterpolantOriginal();
|
|
438
|
-
const interpolate = createdInterpolant.evaluate.bind(createdInterpolant);
|
|
439
|
-
// console.log(interpolate);
|
|
440
|
-
createdInterpolant.evaluate = (time) => {
|
|
441
|
-
const res = interpolate(time);
|
|
442
|
-
quat.set(res[0], res[1], res[2], res[3]);
|
|
443
|
-
quat.premultiply(this._totalOffsetRotation);
|
|
444
|
-
// console.log(new Euler().setFromQuaternion(quat).y.toFixed(2));
|
|
445
|
-
if (trackOffsetRot)
|
|
446
|
-
quat.premultiply(trackOffsetRot);
|
|
447
|
-
if (this.director.animationCallbackReceivers) {
|
|
448
|
-
for (const rec of this.director.animationCallbackReceivers) {
|
|
449
|
-
rec?.onTimelineRotation?.call(rec, this.director, this.target, time, quat);
|
|
450
|
-
}
|
|
451
|
-
}
|
|
452
|
-
res[0] = quat.x;
|
|
453
|
-
res[1] = quat.y;
|
|
454
|
-
res[2] = quat.z;
|
|
455
|
-
res[3] = quat.w;
|
|
456
|
-
return res;
|
|
457
|
-
};
|
|
458
|
-
return createdInterpolant;
|
|
459
|
-
};
|
|
460
|
-
}
|
|
461
|
-
createPositionInterpolant(clip, clipModel, track) {
|
|
462
|
-
const createInterpolantOriginal = track.createInterpolant.bind(track);
|
|
463
|
-
const currentPosition = new Vector3();
|
|
464
|
-
this.ensureTrackOffsets();
|
|
465
|
-
const trackOffsetRot = this.trackOffset?.rotation;
|
|
466
|
-
const trackOffsetPos = this.trackOffset?.position;
|
|
467
|
-
let startOffset = undefined;
|
|
468
|
-
track.createInterpolant = () => {
|
|
469
|
-
const createdInterpolant = createInterpolantOriginal();
|
|
470
|
-
const evaluate = createdInterpolant.evaluate.bind(createdInterpolant);
|
|
471
|
-
createdInterpolant.evaluate = (time) => {
|
|
472
|
-
const res = evaluate(time);
|
|
473
|
-
currentPosition.set(res[0], res[1], res[2]);
|
|
474
|
-
if (clipModel.removeStartOffset) {
|
|
475
|
-
if (startOffset === undefined) {
|
|
476
|
-
startOffset = null;
|
|
477
|
-
startOffset = this._actionOffsets.find(a => a.clip === clip)?.rootStartPosition?.clone();
|
|
478
|
-
}
|
|
479
|
-
else if (startOffset?.isVector3) {
|
|
480
|
-
currentPosition.sub(startOffset);
|
|
481
|
-
}
|
|
482
|
-
}
|
|
483
|
-
currentPosition.applyQuaternion(this._totalOffsetRotation);
|
|
484
|
-
currentPosition.add(this._totalOffsetPosition);
|
|
485
|
-
// apply track offset
|
|
486
|
-
if (trackOffsetRot)
|
|
487
|
-
currentPosition.applyQuaternion(trackOffsetRot);
|
|
488
|
-
if (trackOffsetPos) {
|
|
489
|
-
// flipped unity X
|
|
490
|
-
currentPosition.x -= trackOffsetPos.x;
|
|
491
|
-
currentPosition.y += trackOffsetPos.y;
|
|
492
|
-
currentPosition.z += trackOffsetPos.z;
|
|
493
|
-
}
|
|
494
|
-
if (this.director.animationCallbackReceivers) {
|
|
495
|
-
for (const rec of this.director.animationCallbackReceivers) {
|
|
496
|
-
rec?.onTimelinePosition?.call(rec, this.director, this.target, time, currentPosition);
|
|
497
|
-
}
|
|
498
|
-
}
|
|
499
|
-
res[0] = currentPosition.x;
|
|
500
|
-
res[1] = currentPosition.y;
|
|
501
|
-
res[2] = currentPosition.z;
|
|
502
|
-
return res;
|
|
503
|
-
};
|
|
504
|
-
return createdInterpolant;
|
|
505
|
-
};
|
|
506
|
-
}
|
|
507
|
-
}
|
|
508
|
-
const muteAudioTracks = getParam("mutetimeline");
|
|
509
|
-
export class AudioTrackHandler extends TrackHandler {
|
|
510
|
-
models = [];
|
|
511
|
-
listener;
|
|
512
|
-
audio = [];
|
|
513
|
-
audioContextTimeOffset = [];
|
|
514
|
-
lastTime = 0;
|
|
515
|
-
audioSource;
|
|
516
|
-
_audioLoader = null;
|
|
517
|
-
getAudioFilePath(path) {
|
|
518
|
-
// TODO: this should be the timeline asset location probably which MIGHT be different
|
|
519
|
-
const glbLocation = this.director.sourceId;
|
|
520
|
-
return resolveUrl(glbLocation, path);
|
|
521
|
-
}
|
|
522
|
-
onAllowAudioChanged(allow) {
|
|
523
|
-
for (let i = 0; i < this.models.length; i++) {
|
|
524
|
-
const model = this.models[i];
|
|
525
|
-
const audio = this.audio[i];
|
|
526
|
-
audio.setVolume(allow ? model.asset.volume : 0);
|
|
527
|
-
}
|
|
528
|
-
}
|
|
529
|
-
addModel(model) {
|
|
530
|
-
const audio = new Audio(this.listener);
|
|
531
|
-
this.audio.push(audio);
|
|
532
|
-
this.models.push(model);
|
|
533
|
-
}
|
|
534
|
-
onDisable() {
|
|
535
|
-
for (const audio of this.audio) {
|
|
536
|
-
if (audio.isPlaying)
|
|
537
|
-
audio.stop();
|
|
538
|
-
}
|
|
539
|
-
}
|
|
540
|
-
onDestroy() {
|
|
541
|
-
for (const audio of this.audio) {
|
|
542
|
-
if (audio.source)
|
|
543
|
-
audio?.disconnect();
|
|
544
|
-
}
|
|
545
|
-
this.audio.length = 0;
|
|
546
|
-
// TODO: dispose loaded audio buffers by this track
|
|
547
|
-
}
|
|
548
|
-
onMuteChanged() {
|
|
549
|
-
if (this.muted) {
|
|
550
|
-
for (let i = 0; i < this.audio.length; i++) {
|
|
551
|
-
const audio = this.audio[i];
|
|
552
|
-
if (audio?.isPlaying)
|
|
553
|
-
audio.stop();
|
|
554
|
-
}
|
|
555
|
-
}
|
|
556
|
-
}
|
|
557
|
-
stop() {
|
|
558
|
-
for (let i = 0; i < this.audio.length; i++) {
|
|
559
|
-
const audio = this.audio[i];
|
|
560
|
-
if (audio?.isPlaying)
|
|
561
|
-
audio.stop();
|
|
562
|
-
}
|
|
563
|
-
}
|
|
564
|
-
evaluate(time) {
|
|
565
|
-
if (muteAudioTracks)
|
|
566
|
-
return;
|
|
567
|
-
if (this.track.muted)
|
|
568
|
-
return;
|
|
569
|
-
if (this.director.speed < 0) {
|
|
570
|
-
// Reversed audio playback is currently not supported
|
|
571
|
-
return;
|
|
572
|
-
}
|
|
573
|
-
const isMuted = this.director.context.application.muted;
|
|
574
|
-
// this is just so that we dont hear the very first beat when the audio starts but is muted
|
|
575
|
-
// if we dont add a delay we hear a little bit of the audio before it shuts down
|
|
576
|
-
// MAYBE instead of doing it like this we should connect a custom audio node (or disconnect the output node?)
|
|
577
|
-
const playTimeOffset = isMuted ? .1 : 0;
|
|
578
|
-
for (let i = 0; i < this.models.length; i++) {
|
|
579
|
-
const model = this.models[i];
|
|
580
|
-
const audio = this.audio[i];
|
|
581
|
-
const asset = model.asset;
|
|
582
|
-
// only trigger loading for tracks that are CLOSE to being played
|
|
583
|
-
if ((!audio || !audio.buffer) && this.isInTimeRange(model, time - 1, time + 1)) {
|
|
584
|
-
this.handleAudioLoading(model, audio);
|
|
585
|
-
}
|
|
586
|
-
if (AudioSource.userInteractionRegistered === false)
|
|
587
|
-
continue;
|
|
588
|
-
if (audio === null || !audio.buffer)
|
|
589
|
-
continue;
|
|
590
|
-
audio.playbackRate = this.director.context.time.timeScale * this.director.speed;
|
|
591
|
-
audio.loop = asset.loop;
|
|
592
|
-
if (time >= model.start && time <= model.end && time < this.director.duration) {
|
|
593
|
-
if (this.director.isPlaying == false) {
|
|
594
|
-
if (audio.isPlaying)
|
|
595
|
-
audio.stop();
|
|
596
|
-
if (this.lastTime === time)
|
|
597
|
-
continue;
|
|
598
|
-
}
|
|
599
|
-
else if (!audio.isPlaying) {
|
|
600
|
-
audio.offset = model.clipIn + (time - model.start) * model.timeScale;
|
|
601
|
-
audio.play(playTimeOffset);
|
|
602
|
-
}
|
|
603
|
-
else {
|
|
604
|
-
const targetOffset = model.clipIn + (time - model.start) * model.timeScale;
|
|
605
|
-
// seems it's non-trivial to get the right time from audio sources;
|
|
606
|
-
// https://github.com/mrdoob/js/blob/master/src/audio/Audio.js#L170
|
|
607
|
-
const currentTime = audio.context.currentTime - audio["_startedAt"] + audio.offset;
|
|
608
|
-
const diff = Math.abs(targetOffset - currentTime);
|
|
609
|
-
if (diff > 0.3) {
|
|
610
|
-
audio.offset = targetOffset;
|
|
611
|
-
audio.stop();
|
|
612
|
-
audio.play(playTimeOffset);
|
|
613
|
-
}
|
|
614
|
-
}
|
|
615
|
-
let vol = asset.volume;
|
|
616
|
-
if (this.track.volume !== undefined)
|
|
617
|
-
vol *= this.track.volume;
|
|
618
|
-
if (isMuted)
|
|
619
|
-
vol = 0;
|
|
620
|
-
if (model.easeInDuration > 0) {
|
|
621
|
-
const easeIn = Math.min((time - model.start) / model.easeInDuration, 1);
|
|
622
|
-
vol *= easeIn;
|
|
623
|
-
}
|
|
624
|
-
if (model.easeOutDuration > 0) {
|
|
625
|
-
const easeOut = Math.min((model.end - time) / model.easeOutDuration, 1);
|
|
626
|
-
vol *= easeOut;
|
|
627
|
-
}
|
|
628
|
-
audio.setVolume(vol * this.director.weight);
|
|
629
|
-
}
|
|
630
|
-
else {
|
|
631
|
-
if (audio.isPlaying)
|
|
632
|
-
audio.stop();
|
|
633
|
-
}
|
|
634
|
-
}
|
|
635
|
-
this.lastTime = time;
|
|
636
|
-
}
|
|
637
|
-
/** Call to load audio buffer for a specific time in the track. Can be used to preload the timeline audio */
|
|
638
|
-
loadAudio(time, lookAhead = 0, lookBehind = 0) {
|
|
639
|
-
let promises = null;
|
|
640
|
-
const rangeStart = time - lookBehind;
|
|
641
|
-
const rangeEnd = time + lookAhead;
|
|
642
|
-
for (const model of this.models) {
|
|
643
|
-
if (this.isInTimeRange(model, rangeStart, rangeEnd)) {
|
|
644
|
-
const audio = this.audio[this.models.indexOf(model)];
|
|
645
|
-
const promise = this.handleAudioLoading(model, audio);
|
|
646
|
-
if (promise !== null) {
|
|
647
|
-
if (promises === null)
|
|
648
|
-
promises = [];
|
|
649
|
-
promises.push(promise);
|
|
650
|
-
}
|
|
651
|
-
}
|
|
652
|
-
}
|
|
653
|
-
if (promises !== null) {
|
|
654
|
-
return Promise.all(promises);
|
|
655
|
-
}
|
|
656
|
-
return null;
|
|
657
|
-
}
|
|
658
|
-
isInTimeRange(model, start, end) {
|
|
659
|
-
// Range surrounds clip range
|
|
660
|
-
if (start <= model.start && end >= model.end)
|
|
661
|
-
return true;
|
|
662
|
-
// Range start is in clip range
|
|
663
|
-
if (start >= model.start && start <= model.end)
|
|
664
|
-
return true;
|
|
665
|
-
// Range end is in clip range
|
|
666
|
-
if (end >= model.start && end <= model.end)
|
|
667
|
-
return true;
|
|
668
|
-
return false;
|
|
669
|
-
}
|
|
670
|
-
static _audioBuffers = new Map();
|
|
671
|
-
static dispose() {
|
|
672
|
-
AudioTrackHandler._audioBuffers.clear();
|
|
673
|
-
}
|
|
674
|
-
handleAudioLoading(model, audio) {
|
|
675
|
-
if (!this._audioLoader) {
|
|
676
|
-
this._audioLoader = new AudioLoader();
|
|
677
|
-
}
|
|
678
|
-
// TODO: maybe we should cache the loaders / buffers here by path
|
|
679
|
-
const path = this.getAudioFilePath(model.asset.clip);
|
|
680
|
-
if (AudioTrackHandler._audioBuffers.get(path)) {
|
|
681
|
-
const promise = AudioTrackHandler._audioBuffers.get(path);
|
|
682
|
-
promise.then((buffer) => {
|
|
683
|
-
if (buffer)
|
|
684
|
-
audio.setBuffer(buffer);
|
|
685
|
-
});
|
|
686
|
-
return promise;
|
|
687
|
-
}
|
|
688
|
-
if (debug)
|
|
689
|
-
console.warn("LOAD audio track", path, this.director.sourceId);
|
|
690
|
-
const loadingPromise = new Promise((resolve, _reject) => {
|
|
691
|
-
this._audioLoader.load(path, buffer => {
|
|
692
|
-
audio.setBuffer(buffer);
|
|
693
|
-
resolve(buffer);
|
|
694
|
-
}, undefined, err => {
|
|
695
|
-
console.error("Error loading audio", err);
|
|
696
|
-
resolve(null);
|
|
697
|
-
});
|
|
698
|
-
});
|
|
699
|
-
AudioTrackHandler._audioBuffers.set(path, loadingPromise);
|
|
700
|
-
return loadingPromise;
|
|
701
|
-
}
|
|
702
|
-
}
|
|
703
|
-
export class SignalTrackHandler extends TrackHandler {
|
|
704
|
-
models = [];
|
|
705
|
-
didTrigger = [];
|
|
706
|
-
receivers = [];
|
|
707
|
-
// TODO: test when timeline signals are being reset in Unity
|
|
708
|
-
// onEnable() {
|
|
709
|
-
// for (let i = 0; i < this.didTrigger?.length; i++) {
|
|
710
|
-
// this.didTrigger[i] = false;
|
|
711
|
-
// }
|
|
712
|
-
// }
|
|
713
|
-
// private _lastTime: number = -1;
|
|
714
|
-
evaluate(time) {
|
|
715
|
-
if (this.track.muted)
|
|
716
|
-
return;
|
|
717
|
-
// let lastTime = this._lastTime;
|
|
718
|
-
// if (lastTime === -1) lastTime = time;
|
|
719
|
-
// this._lastTime = time;
|
|
720
|
-
const estimatedFrameLengthWithPadding = this.director.context.time.deltaTime * 1.5;
|
|
721
|
-
for (let i = 0; i < this.models.length; i++) {
|
|
722
|
-
const model = this.models[i];
|
|
723
|
-
const wasTriggered = this.didTrigger[i];
|
|
724
|
-
const td = model.time - time;
|
|
725
|
-
let isActive = false;
|
|
726
|
-
if (model.retroActive) {
|
|
727
|
-
isActive = td <= 0.000001;
|
|
728
|
-
}
|
|
729
|
-
else {
|
|
730
|
-
const abs = Math.abs(td);
|
|
731
|
-
// e.g. if the signal is at frame 0 and the timeline duration also 0 (no tracks, just a signal at frame 0)
|
|
732
|
-
if (abs === 0) {
|
|
733
|
-
isActive = true;
|
|
734
|
-
}
|
|
735
|
-
else if (abs >= .00001 && abs < estimatedFrameLengthWithPadding) {
|
|
736
|
-
isActive = true;
|
|
737
|
-
}
|
|
738
|
-
}
|
|
739
|
-
// console.log(time, td, isActive);
|
|
740
|
-
if (isActive) {
|
|
741
|
-
if (!wasTriggered) {
|
|
742
|
-
if (debug)
|
|
743
|
-
console.log("Trigger signal", time, model.time, model);
|
|
744
|
-
this.didTrigger[i] = true;
|
|
745
|
-
// If a signal doesnt have any explicit receivers it will invoke the signal globally
|
|
746
|
-
if (this.receivers?.length <= 0) {
|
|
747
|
-
SignalReceiver.invoke(model.asset);
|
|
748
|
-
}
|
|
749
|
-
else {
|
|
750
|
-
for (const rec of this.receivers) {
|
|
751
|
-
if (!rec)
|
|
752
|
-
continue;
|
|
753
|
-
rec.invoke(model.asset);
|
|
754
|
-
}
|
|
755
|
-
}
|
|
756
|
-
}
|
|
757
|
-
}
|
|
758
|
-
else {
|
|
759
|
-
if (!model.emitOnce)
|
|
760
|
-
this.didTrigger[i] = false;
|
|
761
|
-
}
|
|
762
|
-
}
|
|
763
|
-
}
|
|
764
|
-
}
|
|
765
|
-
export class ControlTrackHandler extends TrackHandler {
|
|
766
|
-
models = [];
|
|
767
|
-
timelines = [];
|
|
768
|
-
resolveSourceObjects(_context) {
|
|
769
|
-
for (let i = this.models.length - 1; i >= 0; i--) {
|
|
770
|
-
const model = this.models[i];
|
|
771
|
-
const asset = model.asset;
|
|
772
|
-
if (!asset.sourceObject || typeof asset.sourceObject !== "object") {
|
|
773
|
-
console.log("no source object, removing model", i, asset);
|
|
774
|
-
this.models.splice(i, 1);
|
|
775
|
-
continue;
|
|
776
|
-
}
|
|
777
|
-
else {
|
|
778
|
-
const timeline = GameObject.getComponent(asset.sourceObject, PlayableDirector);
|
|
779
|
-
// always add it to keep size of timelines and models in sync (index of model is index of timeline)
|
|
780
|
-
this.timelines.push(timeline);
|
|
781
|
-
if (timeline) {
|
|
782
|
-
if (asset.updateDirector) {
|
|
783
|
-
timeline.playOnAwake = false;
|
|
784
|
-
}
|
|
785
|
-
}
|
|
786
|
-
}
|
|
787
|
-
}
|
|
788
|
-
}
|
|
789
|
-
_previousActiveModel = null;
|
|
790
|
-
evaluate(time) {
|
|
791
|
-
this._previousActiveModel = null;
|
|
792
|
-
for (let i = 0; i < this.models.length; i++) {
|
|
793
|
-
const model = this.models[i];
|
|
794
|
-
const asset = model.asset;
|
|
795
|
-
if (time >= model.start && time <= model.end) {
|
|
796
|
-
this._previousActiveModel = model;
|
|
797
|
-
const clipTime = this.getClipTime(time, model);
|
|
798
|
-
if (asset.controlActivation) {
|
|
799
|
-
const obj = asset.sourceObject;
|
|
800
|
-
obj.visible = true;
|
|
801
|
-
}
|
|
802
|
-
if (asset.updateDirector) {
|
|
803
|
-
const timeline = this.timelines[i];
|
|
804
|
-
if (timeline) {
|
|
805
|
-
if (timeline.isPlaying) {
|
|
806
|
-
timeline.pause();
|
|
807
|
-
}
|
|
808
|
-
timeline.time = clipTime;
|
|
809
|
-
timeline.evaluate();
|
|
810
|
-
}
|
|
811
|
-
}
|
|
812
|
-
// control tracks can not overlap/blend
|
|
813
|
-
// break;
|
|
814
|
-
}
|
|
815
|
-
else {
|
|
816
|
-
const previousActiveAsset = this._previousActiveModel?.asset;
|
|
817
|
-
if (asset.controlActivation) {
|
|
818
|
-
const obj = asset.sourceObject;
|
|
819
|
-
if (previousActiveAsset?.sourceObject !== obj)
|
|
820
|
-
obj.visible = false;
|
|
821
|
-
}
|
|
822
|
-
}
|
|
823
|
-
}
|
|
824
|
-
}
|
|
825
|
-
}
|
|
1
|
+
import { PlayableDirector } from "./PlayableDirector.js";
|
|
2
|
+
import * as Models from "./TimelineModels.js";
|
|
3
|
+
import { GameObject } from "../Component.js";
|
|
4
|
+
import { SignalReceiver } from "./SignalAsset.js";
|
|
5
|
+
import { Audio, AudioLoader, Euler, Quaternion, QuaternionKeyframeTrack, Vector3, VectorKeyframeTrack } from "three";
|
|
6
|
+
import { getParam, resolveUrl } from "../../engine/engine_utils.js";
|
|
7
|
+
import { AudioSource } from "../AudioSource.js";
|
|
8
|
+
import { Animator } from "../Animator.js";
|
|
9
|
+
import { setObjectAnimated } from "../AnimationUtils.js";
|
|
10
|
+
import { isDevEnvironment } from "../../engine/debug/index.js";
|
|
11
|
+
const debug = getParam("debugtimeline");
|
|
12
|
+
export class TrackHandler {
|
|
13
|
+
director;
|
|
14
|
+
track;
|
|
15
|
+
get muted() { return this.track.muted; }
|
|
16
|
+
set muted(val) {
|
|
17
|
+
if (val !== this.track.muted) {
|
|
18
|
+
this.track.muted = val;
|
|
19
|
+
this.onMuteChanged?.call(this);
|
|
20
|
+
}
|
|
21
|
+
}
|
|
22
|
+
*forEachClip(backwards = false) {
|
|
23
|
+
if (!this.track?.clips)
|
|
24
|
+
return;
|
|
25
|
+
if (backwards) {
|
|
26
|
+
for (let i = this.track.clips.length - 1; i >= 0; i--) {
|
|
27
|
+
yield this.track.clips[i];
|
|
28
|
+
}
|
|
29
|
+
}
|
|
30
|
+
else {
|
|
31
|
+
for (const clip of this.track.clips) {
|
|
32
|
+
yield clip;
|
|
33
|
+
}
|
|
34
|
+
}
|
|
35
|
+
}
|
|
36
|
+
getClipTime(time, model) {
|
|
37
|
+
return model.clipIn + (time - model.start) * model.timeScale;
|
|
38
|
+
}
|
|
39
|
+
getClipTimeNormalized(time, model) {
|
|
40
|
+
return (time - model.start) / model.duration;
|
|
41
|
+
}
|
|
42
|
+
evaluateWeight(time, index, models, isActive = true) {
|
|
43
|
+
if (index < 0 || index >= models.length)
|
|
44
|
+
return 0;
|
|
45
|
+
const model = models[index];
|
|
46
|
+
if (isActive || time >= model.start && time <= model.end) {
|
|
47
|
+
let weight = 1;
|
|
48
|
+
const isBlendingWithNext = false;
|
|
49
|
+
// this blending with next clips is already baked into easeIn/easeOut
|
|
50
|
+
// if (allowBlendWithNext && index + 1 < models.length) {
|
|
51
|
+
// const next = models[index + 1];
|
|
52
|
+
// const nextWeight = (time - next.start) / (model.end - next.start);
|
|
53
|
+
// isBlendingWithNext = nextWeight > 0;
|
|
54
|
+
// weight = 1 - nextWeight;
|
|
55
|
+
// }
|
|
56
|
+
if (model.easeInDuration > 0) {
|
|
57
|
+
const easeIn = Math.min((time - model.start) / model.easeInDuration, 1);
|
|
58
|
+
weight *= easeIn;
|
|
59
|
+
}
|
|
60
|
+
if (model.easeOutDuration > 0 && !isBlendingWithNext) {
|
|
61
|
+
const easeOut = Math.min((model.end - time) / model.easeOutDuration, 1);
|
|
62
|
+
weight *= easeOut;
|
|
63
|
+
}
|
|
64
|
+
return weight;
|
|
65
|
+
}
|
|
66
|
+
return 0;
|
|
67
|
+
}
|
|
68
|
+
}
|
|
69
|
+
class AnimationClipOffsetData {
|
|
70
|
+
clip;
|
|
71
|
+
rootPositionOffset;
|
|
72
|
+
rootQuaternionOffset;
|
|
73
|
+
get hasOffsets() { return this.rootPositionOffset !== undefined || this.rootQuaternionOffset !== undefined; }
|
|
74
|
+
// not necessary
|
|
75
|
+
rootStartPosition;
|
|
76
|
+
rootEndPosition;
|
|
77
|
+
rootStartQuaternion;
|
|
78
|
+
rootEndQuaternion;
|
|
79
|
+
constructor(action) {
|
|
80
|
+
const clip = action.getClip();
|
|
81
|
+
this.clip = clip;
|
|
82
|
+
const root = action.getRoot();
|
|
83
|
+
const rootPositionTrackName = root.name + ".position";
|
|
84
|
+
const rootRotationTrackName = root.name + ".quaternion";
|
|
85
|
+
if (debug)
|
|
86
|
+
console.log(clip.name, clip.tracks, rootPositionTrackName);
|
|
87
|
+
for (const track of clip.tracks) {
|
|
88
|
+
if (track.times.length <= 0)
|
|
89
|
+
continue;
|
|
90
|
+
if (track.name.endsWith(rootPositionTrackName)) {
|
|
91
|
+
this.rootStartPosition = new Vector3().fromArray(track.values, 0);
|
|
92
|
+
this.rootEndPosition = new Vector3().fromArray(track.values, track.values.length - 3);
|
|
93
|
+
this.rootPositionOffset = this.rootEndPosition.clone().sub(this.rootStartPosition);
|
|
94
|
+
if (debug)
|
|
95
|
+
console.log(this.rootPositionOffset);
|
|
96
|
+
// this.rootPositionOffset.set(0, 0, 0);
|
|
97
|
+
}
|
|
98
|
+
else if (track.name.endsWith(rootRotationTrackName)) {
|
|
99
|
+
this.rootStartQuaternion = new Quaternion().fromArray(track.values, 0);
|
|
100
|
+
this.rootEndQuaternion = new Quaternion().fromArray(track.values, track.values.length - 4);
|
|
101
|
+
this.rootQuaternionOffset = this.rootEndQuaternion.clone().multiply(this.rootStartQuaternion);
|
|
102
|
+
if (debug) {
|
|
103
|
+
const euler = new Euler().setFromQuaternion(this.rootQuaternionOffset);
|
|
104
|
+
console.log("ROT", euler);
|
|
105
|
+
}
|
|
106
|
+
}
|
|
107
|
+
}
|
|
108
|
+
}
|
|
109
|
+
}
|
|
110
|
+
// TODO: add support for clip clamp modes (loop, pingpong, clamp)
|
|
111
|
+
export class AnimationTrackHandler extends TrackHandler {
|
|
112
|
+
models = [];
|
|
113
|
+
trackOffset;
|
|
114
|
+
target;
|
|
115
|
+
/** The AnimationMixer, should be shared with the animator if an animator is bound */
|
|
116
|
+
mixer;
|
|
117
|
+
clips = [];
|
|
118
|
+
actions = [];
|
|
119
|
+
/** holds data/info about clips differences */
|
|
120
|
+
_actionOffsets = [];
|
|
121
|
+
_didBind = false;
|
|
122
|
+
_animator = null;
|
|
123
|
+
// Using this callback instead of onEnable etc
|
|
124
|
+
// because we want to re-enable the animator when the director is at the end and wrap mode is set to none
|
|
125
|
+
// in which case the director is stopped (but not disabled)
|
|
126
|
+
// which means we want to notify the object that it's not animated anymore
|
|
127
|
+
// and the animator can then take over
|
|
128
|
+
onStateChanged() {
|
|
129
|
+
if (this._animator)
|
|
130
|
+
setObjectAnimated(this._animator.gameObject, this, this.director.isPlaying);
|
|
131
|
+
}
|
|
132
|
+
createHooks(clipModel, clip) {
|
|
133
|
+
if (clip.tracks?.length <= 0) {
|
|
134
|
+
console.warn("No tracks in AnimationClip", clip);
|
|
135
|
+
return;
|
|
136
|
+
}
|
|
137
|
+
// we only want to hook into the binding of the root object
|
|
138
|
+
// TODO: test with a clip with multiple roots
|
|
139
|
+
const parts = clip.tracks[0].name.split(".");
|
|
140
|
+
const rootName = parts[parts.length - 2];
|
|
141
|
+
const positionTrackName = rootName + ".position";
|
|
142
|
+
const rotationTrackName = rootName + ".quaternion";
|
|
143
|
+
let foundPositionTrack = false;
|
|
144
|
+
let foundRotationTrack = false;
|
|
145
|
+
for (const t of clip.tracks) {
|
|
146
|
+
if (t.name.endsWith(positionTrackName)) {
|
|
147
|
+
foundPositionTrack = true;
|
|
148
|
+
this.createPositionInterpolant(clip, clipModel, t);
|
|
149
|
+
}
|
|
150
|
+
else if (t.name.endsWith(rotationTrackName)) {
|
|
151
|
+
foundRotationTrack = true;
|
|
152
|
+
this.createRotationInterpolant(clip, clipModel, t);
|
|
153
|
+
}
|
|
154
|
+
}
|
|
155
|
+
// ensure we always have a position and rotation track so we can apply offsets in interpolator
|
|
156
|
+
// TODO: this currently assumes that there is only one root always that has offsets so it only does create the interpolator for the first track which might be incorrect. In general it would probably be better if we would not create additional tracks but apply the offsets for these objects elsewhere!?
|
|
157
|
+
if (!foundPositionTrack || !foundRotationTrack) {
|
|
158
|
+
const root = this.mixer?.getRoot();
|
|
159
|
+
const track = clip.tracks[0];
|
|
160
|
+
const indexOfProperty = track.name.lastIndexOf(".");
|
|
161
|
+
const baseName = track.name.substring(0, indexOfProperty);
|
|
162
|
+
const objName = baseName.substring(baseName.lastIndexOf(".") + 1);
|
|
163
|
+
const targetObj = root.getObjectByName(objName);
|
|
164
|
+
// TODO can't animate unnamed objects which use GUID as name this way, need scene.getObjectByProperty('uuid', objectName);
|
|
165
|
+
// This should be right but needs testing:
|
|
166
|
+
// const parsedPath = PropertyBinding.parseTrackName(track.name);
|
|
167
|
+
// const targetObj = PropertyBinding.findNode(root, parsedPath.nodeName);
|
|
168
|
+
if (targetObj) {
|
|
169
|
+
if (!foundPositionTrack) {
|
|
170
|
+
const trackName = baseName + ".position";
|
|
171
|
+
if (debug)
|
|
172
|
+
console.warn("Create position track", objName, targetObj);
|
|
173
|
+
// apply initial local position so it doesnt get flipped or otherwise changed
|
|
174
|
+
const pos = targetObj.position;
|
|
175
|
+
const track = new VectorKeyframeTrack(trackName, [0, clip.duration], [pos.x, pos.y, pos.z, pos.x, pos.y, pos.z]);
|
|
176
|
+
clip.tracks.push(track);
|
|
177
|
+
this.createPositionInterpolant(clip, clipModel, track);
|
|
178
|
+
}
|
|
179
|
+
else if (!foundRotationTrack) {
|
|
180
|
+
const trackName = clip.tracks[0].name.substring(0, indexOfProperty) + ".quaternion";
|
|
181
|
+
if (debug)
|
|
182
|
+
console.warn("Create quaternion track", objName, targetObj);
|
|
183
|
+
const rot = targetObj.quaternion;
|
|
184
|
+
const track = new QuaternionKeyframeTrack(trackName, [0, clip.duration], [rot.x, rot.y, rot.z, rot.w, rot.x, rot.y, rot.z, rot.w]);
|
|
185
|
+
clip.tracks.push(track);
|
|
186
|
+
this.createRotationInterpolant(clip, clipModel, track);
|
|
187
|
+
}
|
|
188
|
+
}
|
|
189
|
+
}
|
|
190
|
+
}
|
|
191
|
+
bind() {
|
|
192
|
+
if (this._didBind)
|
|
193
|
+
return;
|
|
194
|
+
this._didBind = true;
|
|
195
|
+
if (debug)
|
|
196
|
+
console.log(this.models);
|
|
197
|
+
// the object being animated
|
|
198
|
+
if (this.mixer)
|
|
199
|
+
this.target = this.mixer.getRoot();
|
|
200
|
+
else
|
|
201
|
+
console.warn("No mixer was assigned to animation track");
|
|
202
|
+
for (const action of this.actions) {
|
|
203
|
+
const off = new AnimationClipOffsetData(action);
|
|
204
|
+
this._actionOffsets.push(off);
|
|
205
|
+
}
|
|
206
|
+
if (this.target) {
|
|
207
|
+
// We need to disable the animator component in case it also animates
|
|
208
|
+
// which overrides the timeline
|
|
209
|
+
this._animator = GameObject.getComponent(this.target, Animator) ?? null;
|
|
210
|
+
if (this._animator) {
|
|
211
|
+
setObjectAnimated(this._animator.gameObject, this, true);
|
|
212
|
+
}
|
|
213
|
+
}
|
|
214
|
+
// Clip Offsets
|
|
215
|
+
for (const model of this.models) {
|
|
216
|
+
const clipData = model.asset;
|
|
217
|
+
const pos = clipData.position;
|
|
218
|
+
const rot = clipData.rotation;
|
|
219
|
+
if (pos && pos.x !== undefined) {
|
|
220
|
+
if (!pos.isVector3) {
|
|
221
|
+
clipData.position = new Vector3(pos.x, pos.y, pos.z);
|
|
222
|
+
}
|
|
223
|
+
if (!rot.isQuaternion) {
|
|
224
|
+
clipData.rotation = new Quaternion(rot.x, rot.y, rot.z, rot.w);
|
|
225
|
+
}
|
|
226
|
+
}
|
|
227
|
+
}
|
|
228
|
+
this.ensureTrackOffsets();
|
|
229
|
+
}
|
|
230
|
+
ensureTrackOffsets() {
|
|
231
|
+
if (this.trackOffset) {
|
|
232
|
+
const pos = this.trackOffset.position;
|
|
233
|
+
if (pos) {
|
|
234
|
+
if (!pos.isVector3) {
|
|
235
|
+
this.trackOffset.position = new Vector3(pos.x, pos.y, pos.z);
|
|
236
|
+
}
|
|
237
|
+
}
|
|
238
|
+
const rot = this.trackOffset.rotation;
|
|
239
|
+
if (rot) {
|
|
240
|
+
if (!rot.isQuaternion) {
|
|
241
|
+
this.trackOffset.rotation = new Quaternion(rot.x, rot.y, rot.z, rot.w);
|
|
242
|
+
}
|
|
243
|
+
}
|
|
244
|
+
}
|
|
245
|
+
}
|
|
246
|
+
_useclipOffsets = true;
|
|
247
|
+
_totalOffsetPosition = new Vector3();
|
|
248
|
+
_totalOffsetRotation = new Quaternion();
|
|
249
|
+
_totalOffsetPosition2 = new Vector3();
|
|
250
|
+
_totalOffsetRotation2 = new Quaternion();
|
|
251
|
+
_summedPos = new Vector3();
|
|
252
|
+
_tempPos = new Vector3();
|
|
253
|
+
_summedRot = new Quaternion();
|
|
254
|
+
_tempRot = new Quaternion();
|
|
255
|
+
_clipRotQuat = new Quaternion();
|
|
256
|
+
evaluate(time) {
|
|
257
|
+
if (this.track.muted)
|
|
258
|
+
return;
|
|
259
|
+
if (!this.mixer)
|
|
260
|
+
return;
|
|
261
|
+
this.bind();
|
|
262
|
+
// if (this._animator && this.director.isPlaying && this.director.weight > 0) this._animator.enabled = false;
|
|
263
|
+
this._totalOffsetPosition.set(0, 0, 0);
|
|
264
|
+
this._totalOffsetRotation.set(0, 0, 0, 1);
|
|
265
|
+
this._totalOffsetPosition2.set(0, 0, 0);
|
|
266
|
+
this._totalOffsetRotation2.set(0, 0, 0, 1);
|
|
267
|
+
let activeClips = 0;
|
|
268
|
+
let blend = 0;
|
|
269
|
+
let didPostExtrapolate = false;
|
|
270
|
+
let didPreExtrapolate = false;
|
|
271
|
+
// The total weight is used to blend with the animator controller active states
|
|
272
|
+
let totalWeight = 0;
|
|
273
|
+
for (let i = 0; i < this.clips.length; i++) {
|
|
274
|
+
const model = this.models[i];
|
|
275
|
+
const action = this.actions[i];
|
|
276
|
+
const clipModel = model.asset;
|
|
277
|
+
action.weight = 0;
|
|
278
|
+
const isInTimeRange = time >= model.start && time <= model.end;
|
|
279
|
+
const preExtrapolation = model.preExtrapolationMode;
|
|
280
|
+
const postExtrapolation = model.postExtrapolationMode;
|
|
281
|
+
const nextClip = i < this.clips.length - 1 ? this.models[i + 1] : null;
|
|
282
|
+
let isActive = isInTimeRange;
|
|
283
|
+
let doPreExtrapolate = false;
|
|
284
|
+
if (!isActive && !didPostExtrapolate && model.end < time && postExtrapolation !== Models.ClipExtrapolation.None) {
|
|
285
|
+
// use post extrapolate if its the last clip of the next clip has not yet started
|
|
286
|
+
if (!nextClip || nextClip.start > time) {
|
|
287
|
+
isActive = true;
|
|
288
|
+
didPostExtrapolate = true;
|
|
289
|
+
}
|
|
290
|
+
}
|
|
291
|
+
else if (i == 0 && !isActive && !didPreExtrapolate && model.start > time && preExtrapolation !== Models.ClipExtrapolation.None) {
|
|
292
|
+
if (!nextClip || nextClip.start < time) {
|
|
293
|
+
isActive = true;
|
|
294
|
+
doPreExtrapolate = true;
|
|
295
|
+
didPreExtrapolate = true;
|
|
296
|
+
}
|
|
297
|
+
}
|
|
298
|
+
if (isActive) {
|
|
299
|
+
// const clip = this.clips[i];
|
|
300
|
+
let weight = 1;
|
|
301
|
+
weight *= this.evaluateWeight(time, i, this.models, isActive);
|
|
302
|
+
weight *= this.director.weight;
|
|
303
|
+
let handleLoop = isInTimeRange;
|
|
304
|
+
if (doPreExtrapolate) {
|
|
305
|
+
switch (preExtrapolation) {
|
|
306
|
+
case Models.ClipExtrapolation.Hold:
|
|
307
|
+
// Nothing to do
|
|
308
|
+
break;
|
|
309
|
+
case Models.ClipExtrapolation.Loop:
|
|
310
|
+
// TODO: this is not correct yet
|
|
311
|
+
time += model.start;
|
|
312
|
+
handleLoop = true;
|
|
313
|
+
break;
|
|
314
|
+
default:
|
|
315
|
+
time += model.start;
|
|
316
|
+
handleLoop = true;
|
|
317
|
+
break;
|
|
318
|
+
}
|
|
319
|
+
}
|
|
320
|
+
// TODO: handle clipIn again
|
|
321
|
+
let t = this.getClipTime(time, model);
|
|
322
|
+
let loops = 0;
|
|
323
|
+
const duration = clipModel.duration;
|
|
324
|
+
// This is the actual duration of the clip in the timeline (with clipping and scale)
|
|
325
|
+
// const clipDuration = (model.end - model.start) * model.timeScale;
|
|
326
|
+
if (doPreExtrapolate) {
|
|
327
|
+
if (preExtrapolation === Models.ClipExtrapolation.Hold) {
|
|
328
|
+
t = 0;
|
|
329
|
+
}
|
|
330
|
+
}
|
|
331
|
+
if (handleLoop) {
|
|
332
|
+
if (clipModel.loop) {
|
|
333
|
+
// const t0 = t - .001;
|
|
334
|
+
loops += Math.floor(t / (duration + .000001));
|
|
335
|
+
while (t > duration) {
|
|
336
|
+
t -= duration;
|
|
337
|
+
}
|
|
338
|
+
}
|
|
339
|
+
}
|
|
340
|
+
else if (!isInTimeRange) {
|
|
341
|
+
if (didPostExtrapolate) {
|
|
342
|
+
switch (postExtrapolation) {
|
|
343
|
+
case Models.ClipExtrapolation.Hold:
|
|
344
|
+
t = this.getClipTime(model.end, model);
|
|
345
|
+
break;
|
|
346
|
+
case Models.ClipExtrapolation.Loop:
|
|
347
|
+
t %= duration;
|
|
348
|
+
break;
|
|
349
|
+
case Models.ClipExtrapolation.PingPong:
|
|
350
|
+
const loops = Math.floor(t / duration);
|
|
351
|
+
const invert = loops % 2 !== 0;
|
|
352
|
+
t %= duration;
|
|
353
|
+
if (invert)
|
|
354
|
+
t = duration - t;
|
|
355
|
+
break;
|
|
356
|
+
}
|
|
357
|
+
}
|
|
358
|
+
}
|
|
359
|
+
if (model.reversed === true)
|
|
360
|
+
action.time = action.getClip().duration - t;
|
|
361
|
+
else
|
|
362
|
+
action.time = t;
|
|
363
|
+
action.timeScale = 0;
|
|
364
|
+
const effectiveWeight = Math.max(0, weight);
|
|
365
|
+
action.weight = effectiveWeight;
|
|
366
|
+
totalWeight += effectiveWeight;
|
|
367
|
+
action.clampWhenFinished = false;
|
|
368
|
+
if (!action.isRunning())
|
|
369
|
+
action.play();
|
|
370
|
+
// console.log(action.time, action.weight);
|
|
371
|
+
if (this._useclipOffsets) {
|
|
372
|
+
const totalPosition = activeClips == 0 ? this._totalOffsetPosition : this._totalOffsetPosition2;
|
|
373
|
+
const totalRotation = activeClips == 0 ? this._totalOffsetRotation : this._totalOffsetRotation2;
|
|
374
|
+
if (activeClips < 1)
|
|
375
|
+
blend = 1 - weight;
|
|
376
|
+
activeClips += 1;
|
|
377
|
+
const summedPos = this._summedPos.set(0, 0, 0);
|
|
378
|
+
const tempPos = this._tempPos.set(0, 0, 0);
|
|
379
|
+
const summedRot = this._summedRot.identity();
|
|
380
|
+
const tempRot = this._tempRot.identity();
|
|
381
|
+
const clipOffsetRot = clipModel.rotation;
|
|
382
|
+
if (clipOffsetRot) {
|
|
383
|
+
this._clipRotQuat.identity();
|
|
384
|
+
this._clipRotQuat.slerp(clipOffsetRot, weight);
|
|
385
|
+
}
|
|
386
|
+
const offsets = this._actionOffsets[i];
|
|
387
|
+
if (offsets.hasOffsets) {
|
|
388
|
+
for (let i = 0; i < loops; i++) {
|
|
389
|
+
if (offsets.rootPositionOffset)
|
|
390
|
+
tempPos.copy(offsets.rootPositionOffset);
|
|
391
|
+
else
|
|
392
|
+
tempPos.set(0, 0, 0);
|
|
393
|
+
tempPos.applyQuaternion(summedRot);
|
|
394
|
+
if (this._clipRotQuat)
|
|
395
|
+
tempPos.applyQuaternion(this._clipRotQuat);
|
|
396
|
+
if (offsets.rootQuaternionOffset) {
|
|
397
|
+
// console.log(new Euler().setFromQuaternion(offsets.rootQuaternionOffset).y.toFixed(2));
|
|
398
|
+
tempRot.copy(offsets.rootQuaternionOffset);
|
|
399
|
+
summedRot.multiply(tempRot);
|
|
400
|
+
}
|
|
401
|
+
summedPos.add(tempPos);
|
|
402
|
+
}
|
|
403
|
+
}
|
|
404
|
+
if (this._clipRotQuat)
|
|
405
|
+
totalRotation.multiply(this._clipRotQuat);
|
|
406
|
+
totalRotation.multiply(summedRot);
|
|
407
|
+
if (clipModel.position)
|
|
408
|
+
summedPos.add(clipModel.position);
|
|
409
|
+
totalPosition.add(summedPos);
|
|
410
|
+
}
|
|
411
|
+
}
|
|
412
|
+
}
|
|
413
|
+
if (this._useclipOffsets) {
|
|
414
|
+
this._totalOffsetPosition.lerp(this._totalOffsetPosition2, blend);
|
|
415
|
+
this._totalOffsetRotation.slerp(this._totalOffsetRotation2, blend);
|
|
416
|
+
}
|
|
417
|
+
if (this["__mixerError"] === undefined && (debug || isDevEnvironment()) && this._animator?.runtimeAnimatorController?.mixer && this.mixer !== this._animator?.runtimeAnimatorController?.mixer) {
|
|
418
|
+
this["__mixerError"] = true;
|
|
419
|
+
console.error("AnimationTrack mixer is not shared with the animator controller - this might result in the timeline to not animate properly. Please report a bug to the Needle Engine team!", this);
|
|
420
|
+
}
|
|
421
|
+
if (this._animator?.runtimeAnimatorController) {
|
|
422
|
+
// If the Timeline is running then the timeline track takes control over the animatorcontroller
|
|
423
|
+
// we calculate the weight left for the animatorcontroller actions
|
|
424
|
+
const weightLeft = Math.max(0, 1 - totalWeight);
|
|
425
|
+
this._animator?.runtimeAnimatorController?.update(weightLeft);
|
|
426
|
+
}
|
|
427
|
+
else {
|
|
428
|
+
this.mixer.update(time);
|
|
429
|
+
}
|
|
430
|
+
}
|
|
431
|
+
createRotationInterpolant(_clip, _clipModel, track) {
|
|
432
|
+
const createInterpolantOriginal = track.createInterpolant.bind(track);
|
|
433
|
+
const quat = new Quaternion();
|
|
434
|
+
this.ensureTrackOffsets();
|
|
435
|
+
const trackOffsetRot = this.trackOffset?.rotation;
|
|
436
|
+
track.createInterpolant = () => {
|
|
437
|
+
const createdInterpolant = createInterpolantOriginal();
|
|
438
|
+
const interpolate = createdInterpolant.evaluate.bind(createdInterpolant);
|
|
439
|
+
// console.log(interpolate);
|
|
440
|
+
createdInterpolant.evaluate = (time) => {
|
|
441
|
+
const res = interpolate(time);
|
|
442
|
+
quat.set(res[0], res[1], res[2], res[3]);
|
|
443
|
+
quat.premultiply(this._totalOffsetRotation);
|
|
444
|
+
// console.log(new Euler().setFromQuaternion(quat).y.toFixed(2));
|
|
445
|
+
if (trackOffsetRot)
|
|
446
|
+
quat.premultiply(trackOffsetRot);
|
|
447
|
+
if (this.director.animationCallbackReceivers) {
|
|
448
|
+
for (const rec of this.director.animationCallbackReceivers) {
|
|
449
|
+
rec?.onTimelineRotation?.call(rec, this.director, this.target, time, quat);
|
|
450
|
+
}
|
|
451
|
+
}
|
|
452
|
+
res[0] = quat.x;
|
|
453
|
+
res[1] = quat.y;
|
|
454
|
+
res[2] = quat.z;
|
|
455
|
+
res[3] = quat.w;
|
|
456
|
+
return res;
|
|
457
|
+
};
|
|
458
|
+
return createdInterpolant;
|
|
459
|
+
};
|
|
460
|
+
}
|
|
461
|
+
createPositionInterpolant(clip, clipModel, track) {
|
|
462
|
+
const createInterpolantOriginal = track.createInterpolant.bind(track);
|
|
463
|
+
const currentPosition = new Vector3();
|
|
464
|
+
this.ensureTrackOffsets();
|
|
465
|
+
const trackOffsetRot = this.trackOffset?.rotation;
|
|
466
|
+
const trackOffsetPos = this.trackOffset?.position;
|
|
467
|
+
let startOffset = undefined;
|
|
468
|
+
track.createInterpolant = () => {
|
|
469
|
+
const createdInterpolant = createInterpolantOriginal();
|
|
470
|
+
const evaluate = createdInterpolant.evaluate.bind(createdInterpolant);
|
|
471
|
+
createdInterpolant.evaluate = (time) => {
|
|
472
|
+
const res = evaluate(time);
|
|
473
|
+
currentPosition.set(res[0], res[1], res[2]);
|
|
474
|
+
if (clipModel.removeStartOffset) {
|
|
475
|
+
if (startOffset === undefined) {
|
|
476
|
+
startOffset = null;
|
|
477
|
+
startOffset = this._actionOffsets.find(a => a.clip === clip)?.rootStartPosition?.clone();
|
|
478
|
+
}
|
|
479
|
+
else if (startOffset?.isVector3) {
|
|
480
|
+
currentPosition.sub(startOffset);
|
|
481
|
+
}
|
|
482
|
+
}
|
|
483
|
+
currentPosition.applyQuaternion(this._totalOffsetRotation);
|
|
484
|
+
currentPosition.add(this._totalOffsetPosition);
|
|
485
|
+
// apply track offset
|
|
486
|
+
if (trackOffsetRot)
|
|
487
|
+
currentPosition.applyQuaternion(trackOffsetRot);
|
|
488
|
+
if (trackOffsetPos) {
|
|
489
|
+
// flipped unity X
|
|
490
|
+
currentPosition.x -= trackOffsetPos.x;
|
|
491
|
+
currentPosition.y += trackOffsetPos.y;
|
|
492
|
+
currentPosition.z += trackOffsetPos.z;
|
|
493
|
+
}
|
|
494
|
+
if (this.director.animationCallbackReceivers) {
|
|
495
|
+
for (const rec of this.director.animationCallbackReceivers) {
|
|
496
|
+
rec?.onTimelinePosition?.call(rec, this.director, this.target, time, currentPosition);
|
|
497
|
+
}
|
|
498
|
+
}
|
|
499
|
+
res[0] = currentPosition.x;
|
|
500
|
+
res[1] = currentPosition.y;
|
|
501
|
+
res[2] = currentPosition.z;
|
|
502
|
+
return res;
|
|
503
|
+
};
|
|
504
|
+
return createdInterpolant;
|
|
505
|
+
};
|
|
506
|
+
}
|
|
507
|
+
}
|
|
508
|
+
const muteAudioTracks = getParam("mutetimeline");
|
|
509
|
+
export class AudioTrackHandler extends TrackHandler {
|
|
510
|
+
models = [];
|
|
511
|
+
listener;
|
|
512
|
+
audio = [];
|
|
513
|
+
audioContextTimeOffset = [];
|
|
514
|
+
lastTime = 0;
|
|
515
|
+
audioSource;
|
|
516
|
+
_audioLoader = null;
|
|
517
|
+
getAudioFilePath(path) {
|
|
518
|
+
// TODO: this should be the timeline asset location probably which MIGHT be different
|
|
519
|
+
const glbLocation = this.director.sourceId;
|
|
520
|
+
return resolveUrl(glbLocation, path);
|
|
521
|
+
}
|
|
522
|
+
onAllowAudioChanged(allow) {
|
|
523
|
+
for (let i = 0; i < this.models.length; i++) {
|
|
524
|
+
const model = this.models[i];
|
|
525
|
+
const audio = this.audio[i];
|
|
526
|
+
audio.setVolume(allow ? model.asset.volume : 0);
|
|
527
|
+
}
|
|
528
|
+
}
|
|
529
|
+
addModel(model) {
|
|
530
|
+
const audio = new Audio(this.listener);
|
|
531
|
+
this.audio.push(audio);
|
|
532
|
+
this.models.push(model);
|
|
533
|
+
}
|
|
534
|
+
onDisable() {
|
|
535
|
+
for (const audio of this.audio) {
|
|
536
|
+
if (audio.isPlaying)
|
|
537
|
+
audio.stop();
|
|
538
|
+
}
|
|
539
|
+
}
|
|
540
|
+
onDestroy() {
|
|
541
|
+
for (const audio of this.audio) {
|
|
542
|
+
if (audio.source)
|
|
543
|
+
audio?.disconnect();
|
|
544
|
+
}
|
|
545
|
+
this.audio.length = 0;
|
|
546
|
+
// TODO: dispose loaded audio buffers by this track
|
|
547
|
+
}
|
|
548
|
+
onMuteChanged() {
|
|
549
|
+
if (this.muted) {
|
|
550
|
+
for (let i = 0; i < this.audio.length; i++) {
|
|
551
|
+
const audio = this.audio[i];
|
|
552
|
+
if (audio?.isPlaying)
|
|
553
|
+
audio.stop();
|
|
554
|
+
}
|
|
555
|
+
}
|
|
556
|
+
}
|
|
557
|
+
stop() {
|
|
558
|
+
for (let i = 0; i < this.audio.length; i++) {
|
|
559
|
+
const audio = this.audio[i];
|
|
560
|
+
if (audio?.isPlaying)
|
|
561
|
+
audio.stop();
|
|
562
|
+
}
|
|
563
|
+
}
|
|
564
|
+
evaluate(time) {
|
|
565
|
+
if (muteAudioTracks)
|
|
566
|
+
return;
|
|
567
|
+
if (this.track.muted)
|
|
568
|
+
return;
|
|
569
|
+
if (this.director.speed < 0) {
|
|
570
|
+
// Reversed audio playback is currently not supported
|
|
571
|
+
return;
|
|
572
|
+
}
|
|
573
|
+
const isMuted = this.director.context.application.muted;
|
|
574
|
+
// this is just so that we dont hear the very first beat when the audio starts but is muted
|
|
575
|
+
// if we dont add a delay we hear a little bit of the audio before it shuts down
|
|
576
|
+
// MAYBE instead of doing it like this we should connect a custom audio node (or disconnect the output node?)
|
|
577
|
+
const playTimeOffset = isMuted ? .1 : 0;
|
|
578
|
+
for (let i = 0; i < this.models.length; i++) {
|
|
579
|
+
const model = this.models[i];
|
|
580
|
+
const audio = this.audio[i];
|
|
581
|
+
const asset = model.asset;
|
|
582
|
+
// only trigger loading for tracks that are CLOSE to being played
|
|
583
|
+
if ((!audio || !audio.buffer) && this.isInTimeRange(model, time - 1, time + 1)) {
|
|
584
|
+
this.handleAudioLoading(model, audio);
|
|
585
|
+
}
|
|
586
|
+
if (AudioSource.userInteractionRegistered === false)
|
|
587
|
+
continue;
|
|
588
|
+
if (audio === null || !audio.buffer)
|
|
589
|
+
continue;
|
|
590
|
+
audio.playbackRate = this.director.context.time.timeScale * this.director.speed;
|
|
591
|
+
audio.loop = asset.loop;
|
|
592
|
+
if (time >= model.start && time <= model.end && time < this.director.duration) {
|
|
593
|
+
if (this.director.isPlaying == false) {
|
|
594
|
+
if (audio.isPlaying)
|
|
595
|
+
audio.stop();
|
|
596
|
+
if (this.lastTime === time)
|
|
597
|
+
continue;
|
|
598
|
+
}
|
|
599
|
+
else if (!audio.isPlaying) {
|
|
600
|
+
audio.offset = model.clipIn + (time - model.start) * model.timeScale;
|
|
601
|
+
audio.play(playTimeOffset);
|
|
602
|
+
}
|
|
603
|
+
else {
|
|
604
|
+
const targetOffset = model.clipIn + (time - model.start) * model.timeScale;
|
|
605
|
+
// seems it's non-trivial to get the right time from audio sources;
|
|
606
|
+
// https://github.com/mrdoob/js/blob/master/src/audio/Audio.js#L170
|
|
607
|
+
const currentTime = audio.context.currentTime - audio["_startedAt"] + audio.offset;
|
|
608
|
+
const diff = Math.abs(targetOffset - currentTime);
|
|
609
|
+
if (diff > 0.3) {
|
|
610
|
+
audio.offset = targetOffset;
|
|
611
|
+
audio.stop();
|
|
612
|
+
audio.play(playTimeOffset);
|
|
613
|
+
}
|
|
614
|
+
}
|
|
615
|
+
let vol = asset.volume;
|
|
616
|
+
if (this.track.volume !== undefined)
|
|
617
|
+
vol *= this.track.volume;
|
|
618
|
+
if (isMuted)
|
|
619
|
+
vol = 0;
|
|
620
|
+
if (model.easeInDuration > 0) {
|
|
621
|
+
const easeIn = Math.min((time - model.start) / model.easeInDuration, 1);
|
|
622
|
+
vol *= easeIn;
|
|
623
|
+
}
|
|
624
|
+
if (model.easeOutDuration > 0) {
|
|
625
|
+
const easeOut = Math.min((model.end - time) / model.easeOutDuration, 1);
|
|
626
|
+
vol *= easeOut;
|
|
627
|
+
}
|
|
628
|
+
audio.setVolume(vol * this.director.weight);
|
|
629
|
+
}
|
|
630
|
+
else {
|
|
631
|
+
if (audio.isPlaying)
|
|
632
|
+
audio.stop();
|
|
633
|
+
}
|
|
634
|
+
}
|
|
635
|
+
this.lastTime = time;
|
|
636
|
+
}
|
|
637
|
+
/** Call to load audio buffer for a specific time in the track. Can be used to preload the timeline audio */
|
|
638
|
+
loadAudio(time, lookAhead = 0, lookBehind = 0) {
|
|
639
|
+
let promises = null;
|
|
640
|
+
const rangeStart = time - lookBehind;
|
|
641
|
+
const rangeEnd = time + lookAhead;
|
|
642
|
+
for (const model of this.models) {
|
|
643
|
+
if (this.isInTimeRange(model, rangeStart, rangeEnd)) {
|
|
644
|
+
const audio = this.audio[this.models.indexOf(model)];
|
|
645
|
+
const promise = this.handleAudioLoading(model, audio);
|
|
646
|
+
if (promise !== null) {
|
|
647
|
+
if (promises === null)
|
|
648
|
+
promises = [];
|
|
649
|
+
promises.push(promise);
|
|
650
|
+
}
|
|
651
|
+
}
|
|
652
|
+
}
|
|
653
|
+
if (promises !== null) {
|
|
654
|
+
return Promise.all(promises);
|
|
655
|
+
}
|
|
656
|
+
return null;
|
|
657
|
+
}
|
|
658
|
+
isInTimeRange(model, start, end) {
|
|
659
|
+
// Range surrounds clip range
|
|
660
|
+
if (start <= model.start && end >= model.end)
|
|
661
|
+
return true;
|
|
662
|
+
// Range start is in clip range
|
|
663
|
+
if (start >= model.start && start <= model.end)
|
|
664
|
+
return true;
|
|
665
|
+
// Range end is in clip range
|
|
666
|
+
if (end >= model.start && end <= model.end)
|
|
667
|
+
return true;
|
|
668
|
+
return false;
|
|
669
|
+
}
|
|
670
|
+
static _audioBuffers = new Map();
|
|
671
|
+
static dispose() {
|
|
672
|
+
AudioTrackHandler._audioBuffers.clear();
|
|
673
|
+
}
|
|
674
|
+
handleAudioLoading(model, audio) {
|
|
675
|
+
if (!this._audioLoader) {
|
|
676
|
+
this._audioLoader = new AudioLoader();
|
|
677
|
+
}
|
|
678
|
+
// TODO: maybe we should cache the loaders / buffers here by path
|
|
679
|
+
const path = this.getAudioFilePath(model.asset.clip);
|
|
680
|
+
if (AudioTrackHandler._audioBuffers.get(path)) {
|
|
681
|
+
const promise = AudioTrackHandler._audioBuffers.get(path);
|
|
682
|
+
promise.then((buffer) => {
|
|
683
|
+
if (buffer)
|
|
684
|
+
audio.setBuffer(buffer);
|
|
685
|
+
});
|
|
686
|
+
return promise;
|
|
687
|
+
}
|
|
688
|
+
if (debug)
|
|
689
|
+
console.warn("LOAD audio track", path, this.director.sourceId);
|
|
690
|
+
const loadingPromise = new Promise((resolve, _reject) => {
|
|
691
|
+
this._audioLoader.load(path, buffer => {
|
|
692
|
+
audio.setBuffer(buffer);
|
|
693
|
+
resolve(buffer);
|
|
694
|
+
}, undefined, err => {
|
|
695
|
+
console.error("Error loading audio", err);
|
|
696
|
+
resolve(null);
|
|
697
|
+
});
|
|
698
|
+
});
|
|
699
|
+
AudioTrackHandler._audioBuffers.set(path, loadingPromise);
|
|
700
|
+
return loadingPromise;
|
|
701
|
+
}
|
|
702
|
+
}
|
|
703
|
+
export class SignalTrackHandler extends TrackHandler {
|
|
704
|
+
models = [];
|
|
705
|
+
didTrigger = [];
|
|
706
|
+
receivers = [];
|
|
707
|
+
// TODO: test when timeline signals are being reset in Unity
|
|
708
|
+
// onEnable() {
|
|
709
|
+
// for (let i = 0; i < this.didTrigger?.length; i++) {
|
|
710
|
+
// this.didTrigger[i] = false;
|
|
711
|
+
// }
|
|
712
|
+
// }
|
|
713
|
+
// private _lastTime: number = -1;
|
|
714
|
+
evaluate(time) {
|
|
715
|
+
if (this.track.muted)
|
|
716
|
+
return;
|
|
717
|
+
// let lastTime = this._lastTime;
|
|
718
|
+
// if (lastTime === -1) lastTime = time;
|
|
719
|
+
// this._lastTime = time;
|
|
720
|
+
const estimatedFrameLengthWithPadding = this.director.context.time.deltaTime * 1.5;
|
|
721
|
+
for (let i = 0; i < this.models.length; i++) {
|
|
722
|
+
const model = this.models[i];
|
|
723
|
+
const wasTriggered = this.didTrigger[i];
|
|
724
|
+
const td = model.time - time;
|
|
725
|
+
let isActive = false;
|
|
726
|
+
if (model.retroActive) {
|
|
727
|
+
isActive = td <= 0.000001;
|
|
728
|
+
}
|
|
729
|
+
else {
|
|
730
|
+
const abs = Math.abs(td);
|
|
731
|
+
// e.g. if the signal is at frame 0 and the timeline duration also 0 (no tracks, just a signal at frame 0)
|
|
732
|
+
if (abs === 0) {
|
|
733
|
+
isActive = true;
|
|
734
|
+
}
|
|
735
|
+
else if (abs >= .00001 && abs < estimatedFrameLengthWithPadding) {
|
|
736
|
+
isActive = true;
|
|
737
|
+
}
|
|
738
|
+
}
|
|
739
|
+
// console.log(time, td, isActive);
|
|
740
|
+
if (isActive) {
|
|
741
|
+
if (!wasTriggered) {
|
|
742
|
+
if (debug)
|
|
743
|
+
console.log("Trigger signal", time, model.time, model);
|
|
744
|
+
this.didTrigger[i] = true;
|
|
745
|
+
// If a signal doesnt have any explicit receivers it will invoke the signal globally
|
|
746
|
+
if (this.receivers?.length <= 0) {
|
|
747
|
+
SignalReceiver.invoke(model.asset);
|
|
748
|
+
}
|
|
749
|
+
else {
|
|
750
|
+
for (const rec of this.receivers) {
|
|
751
|
+
if (!rec)
|
|
752
|
+
continue;
|
|
753
|
+
rec.invoke(model.asset);
|
|
754
|
+
}
|
|
755
|
+
}
|
|
756
|
+
}
|
|
757
|
+
}
|
|
758
|
+
else {
|
|
759
|
+
if (!model.emitOnce)
|
|
760
|
+
this.didTrigger[i] = false;
|
|
761
|
+
}
|
|
762
|
+
}
|
|
763
|
+
}
|
|
764
|
+
}
|
|
765
|
+
export class ControlTrackHandler extends TrackHandler {
|
|
766
|
+
models = [];
|
|
767
|
+
timelines = [];
|
|
768
|
+
resolveSourceObjects(_context) {
|
|
769
|
+
for (let i = this.models.length - 1; i >= 0; i--) {
|
|
770
|
+
const model = this.models[i];
|
|
771
|
+
const asset = model.asset;
|
|
772
|
+
if (!asset.sourceObject || typeof asset.sourceObject !== "object") {
|
|
773
|
+
console.log("no source object, removing model", i, asset);
|
|
774
|
+
this.models.splice(i, 1);
|
|
775
|
+
continue;
|
|
776
|
+
}
|
|
777
|
+
else {
|
|
778
|
+
const timeline = GameObject.getComponent(asset.sourceObject, PlayableDirector);
|
|
779
|
+
// always add it to keep size of timelines and models in sync (index of model is index of timeline)
|
|
780
|
+
this.timelines.push(timeline);
|
|
781
|
+
if (timeline) {
|
|
782
|
+
if (asset.updateDirector) {
|
|
783
|
+
timeline.playOnAwake = false;
|
|
784
|
+
}
|
|
785
|
+
}
|
|
786
|
+
}
|
|
787
|
+
}
|
|
788
|
+
}
|
|
789
|
+
_previousActiveModel = null;
|
|
790
|
+
evaluate(time) {
|
|
791
|
+
this._previousActiveModel = null;
|
|
792
|
+
for (let i = 0; i < this.models.length; i++) {
|
|
793
|
+
const model = this.models[i];
|
|
794
|
+
const asset = model.asset;
|
|
795
|
+
if (time >= model.start && time <= model.end) {
|
|
796
|
+
this._previousActiveModel = model;
|
|
797
|
+
const clipTime = this.getClipTime(time, model);
|
|
798
|
+
if (asset.controlActivation) {
|
|
799
|
+
const obj = asset.sourceObject;
|
|
800
|
+
obj.visible = true;
|
|
801
|
+
}
|
|
802
|
+
if (asset.updateDirector) {
|
|
803
|
+
const timeline = this.timelines[i];
|
|
804
|
+
if (timeline) {
|
|
805
|
+
if (timeline.isPlaying) {
|
|
806
|
+
timeline.pause();
|
|
807
|
+
}
|
|
808
|
+
timeline.time = clipTime;
|
|
809
|
+
timeline.evaluate();
|
|
810
|
+
}
|
|
811
|
+
}
|
|
812
|
+
// control tracks can not overlap/blend
|
|
813
|
+
// break;
|
|
814
|
+
}
|
|
815
|
+
else {
|
|
816
|
+
const previousActiveAsset = this._previousActiveModel?.asset;
|
|
817
|
+
if (asset.controlActivation) {
|
|
818
|
+
const obj = asset.sourceObject;
|
|
819
|
+
if (previousActiveAsset?.sourceObject !== obj)
|
|
820
|
+
obj.visible = false;
|
|
821
|
+
}
|
|
822
|
+
}
|
|
823
|
+
}
|
|
824
|
+
}
|
|
825
|
+
}
|
|
826
826
|
//# sourceMappingURL=TimelineTracks.js.map
|