angular-three-rapier 2.2.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -0,0 +1,185 @@
1
+ import { ChangeDetectionStrategy, Component, computed, CUSTOM_ELEMENTS_SCHEMA, effect, ElementRef, inject, input, untracked, viewChild, viewChildren, } from '@angular/core';
2
+ import { extend, getLocalState, pick } from 'angular-three';
3
+ import { mergeInputs } from 'ngxtension/inject-inputs';
4
+ import { DynamicDrawUsage, Object3D } from 'three';
5
+ import { NgtrPhysics } from './physics';
6
+ import { NgtrAnyCollider, NgtrRigidBody, rigidBodyDefaultOptions } from './rigid-body';
7
+ import { createColliderOptions } from './utils';
8
+ import * as i0 from "@angular/core";
9
+ const defaultOptions = rigidBodyDefaultOptions;
10
+ export class NgtrInstancedRigidBodies {
11
+ position = input([0, 0, 0]);
12
+ rotation = input([0, 0, 0]);
13
+ scale = input([1, 1, 1]);
14
+ quaternion = input([0, 0, 0, 1]);
15
+ userData = input({});
16
+ instances = input([], {
17
+ alias: 'ngtrInstancedRigidBodies',
18
+ transform: (value) => {
19
+ if (value === '')
20
+ return [];
21
+ return value;
22
+ },
23
+ });
24
+ options = input(defaultOptions, { transform: mergeInputs(defaultOptions) });
25
+ instanceWrapperRef = viewChild.required('instanceWrapper');
26
+ rigidBodyRefs = viewChildren(NgtrRigidBody);
27
+ physics = inject(NgtrPhysics);
28
+ objectRef = inject(ElementRef);
29
+ colliders = pick(this.options, 'colliders');
30
+ instancedMesh = computed(() => {
31
+ const instanceWrapper = this.instanceWrapperRef().nativeElement;
32
+ if (!instanceWrapper)
33
+ return null;
34
+ const localState = getLocalState(instanceWrapper);
35
+ if (!localState)
36
+ return null;
37
+ // track object's children
38
+ localState.objects();
39
+ const firstChild = instanceWrapper.children[0];
40
+ if (!firstChild || !firstChild.isInstancedMesh)
41
+ return null;
42
+ return firstChild;
43
+ });
44
+ instancesOptions = computed(() => {
45
+ const [instances, options, instancedMesh] = [this.instances(), untracked(this.options), this.instancedMesh()];
46
+ if (!instancedMesh)
47
+ return [];
48
+ return instances.map((instance, index) => ({
49
+ ...instance,
50
+ options: {
51
+ ...options,
52
+ ...(instance.options || {}),
53
+ transformState: (state) => {
54
+ return {
55
+ ...state,
56
+ getMatrix: (matrix) => {
57
+ instancedMesh.getMatrixAt(index, matrix);
58
+ return matrix;
59
+ },
60
+ setMatrix: (matrix) => {
61
+ instancedMesh.setMatrixAt(index, matrix);
62
+ instancedMesh.instanceMatrix.needsUpdate = true;
63
+ },
64
+ meshType: 'instancedMesh',
65
+ };
66
+ },
67
+ },
68
+ key: `${instance.key}-${index}` + `${instancedMesh?.uuid || ''}`,
69
+ }));
70
+ });
71
+ childColliderOptions = computed(() => {
72
+ const colliders = this.colliders();
73
+ // if self colliders is false explicitly, disable auto colliders for this object entirely.
74
+ if (colliders === false)
75
+ return [];
76
+ const physicsColliders = this.physics.colliders();
77
+ // if physics colliders is false explicitly, disable auto colliders for this object entirely.
78
+ if (physicsColliders === false)
79
+ return [];
80
+ const options = this.options();
81
+ // if colliders on object is not set, use physics colliders
82
+ if (!options.colliders)
83
+ options.colliders = physicsColliders;
84
+ const objectLocalState = getLocalState(this.objectRef.nativeElement);
85
+ // track object's children
86
+ objectLocalState?.nonObjects();
87
+ return createColliderOptions(this.objectRef.nativeElement, options);
88
+ });
89
+ constructor() {
90
+ extend({ Object3D });
91
+ effect(() => {
92
+ this.setInstancedMeshMatrixEffect();
93
+ });
94
+ }
95
+ setInstancedMeshMatrixEffect() {
96
+ const instancedMesh = this.instancedMesh();
97
+ if (!instancedMesh)
98
+ return;
99
+ instancedMesh.instanceMatrix.setUsage(DynamicDrawUsage);
100
+ }
101
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.0", ngImport: i0, type: NgtrInstancedRigidBodies, deps: [], target: i0.ɵɵFactoryTarget.Component });
102
+ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.0", type: NgtrInstancedRigidBodies, isStandalone: true, selector: "ngt-object3D[ngtrInstancedRigidBodies]", inputs: { position: { classPropertyName: "position", publicName: "position", isSignal: true, isRequired: false, transformFunction: null }, rotation: { classPropertyName: "rotation", publicName: "rotation", isSignal: true, isRequired: false, transformFunction: null }, scale: { classPropertyName: "scale", publicName: "scale", isSignal: true, isRequired: false, transformFunction: null }, quaternion: { classPropertyName: "quaternion", publicName: "quaternion", isSignal: true, isRequired: false, transformFunction: null }, userData: { classPropertyName: "userData", publicName: "userData", isSignal: true, isRequired: false, transformFunction: null }, instances: { classPropertyName: "instances", publicName: "ngtrInstancedRigidBodies", isSignal: true, isRequired: false, transformFunction: null }, options: { classPropertyName: "options", publicName: "options", isSignal: true, isRequired: false, transformFunction: null } }, host: { properties: { "position": "position()", "rotation": "rotation()", "scale": "scale()", "quaternion": "quaternion()", "userData": "userData()" } }, viewQueries: [{ propertyName: "instanceWrapperRef", first: true, predicate: ["instanceWrapper"], descendants: true, isSignal: true }, { propertyName: "rigidBodyRefs", predicate: NgtrRigidBody, descendants: true, isSignal: true }], exportAs: ["instancedRigidBodies"], ngImport: i0, template: `
103
+ <ngt-object3D #instanceWrapper>
104
+ <ng-content />
105
+ </ngt-object3D>
106
+
107
+ @for (instance of instancesOptions(); track instance.key) {
108
+ <ngt-object3D
109
+ [ngtrRigidBody]="instance.type"
110
+ [options]="instance.options"
111
+ [position]="instance.position"
112
+ [rotation]="instance.rotation"
113
+ [scale]="instance.scale"
114
+ [quaternion]="instance.quaternion"
115
+ [userData]="instance.userData"
116
+ [name]="instance.key + '-instanced-rigid-body-' + $index"
117
+ >
118
+ <ng-content select="[data-colliders]" />
119
+
120
+ @for (childColliderOption of childColliderOptions(); track $index) {
121
+ <ngt-object3D
122
+ [ngtrCollider]="childColliderOption.shape"
123
+ [args]="childColliderOption.args"
124
+ [position]="childColliderOption.position"
125
+ [rotation]="childColliderOption.rotation"
126
+ [scale]="childColliderOption.scale"
127
+ [name]="objectRef.nativeElement.name + '-instanced-collider-' + $index"
128
+ [options]="childColliderOption.colliderOptions"
129
+ />
130
+ }
131
+ </ngt-object3D>
132
+ }
133
+ `, isInline: true, dependencies: [{ kind: "component", type: NgtrRigidBody, selector: "ngt-object3D[ngtrRigidBody]", inputs: ["ngtrRigidBody", "position", "rotation", "scale", "quaternion", "userData", "options"], outputs: ["wake", "sleep", "collisionEnter", "collisionExit", "intersectionEnter", "intersectionExit", "contactForce"], exportAs: ["rigidBody"] }, { kind: "directive", type: NgtrAnyCollider, selector: "ngt-object3D[ngtrCollider]", inputs: ["position", "rotation", "scale", "quaternion", "userData", "name", "options", "ngtrCollider", "args"], outputs: ["ngtrColliderChange", "argsChange", "collisionEnter", "collisionExit", "intersectionEnter", "intersectionExit", "contactForce"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
134
+ }
135
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.0", ngImport: i0, type: NgtrInstancedRigidBodies, decorators: [{
136
+ type: Component,
137
+ args: [{
138
+ selector: 'ngt-object3D[ngtrInstancedRigidBodies]',
139
+ exportAs: 'instancedRigidBodies',
140
+ standalone: true,
141
+ template: `
142
+ <ngt-object3D #instanceWrapper>
143
+ <ng-content />
144
+ </ngt-object3D>
145
+
146
+ @for (instance of instancesOptions(); track instance.key) {
147
+ <ngt-object3D
148
+ [ngtrRigidBody]="instance.type"
149
+ [options]="instance.options"
150
+ [position]="instance.position"
151
+ [rotation]="instance.rotation"
152
+ [scale]="instance.scale"
153
+ [quaternion]="instance.quaternion"
154
+ [userData]="instance.userData"
155
+ [name]="instance.key + '-instanced-rigid-body-' + $index"
156
+ >
157
+ <ng-content select="[data-colliders]" />
158
+
159
+ @for (childColliderOption of childColliderOptions(); track $index) {
160
+ <ngt-object3D
161
+ [ngtrCollider]="childColliderOption.shape"
162
+ [args]="childColliderOption.args"
163
+ [position]="childColliderOption.position"
164
+ [rotation]="childColliderOption.rotation"
165
+ [scale]="childColliderOption.scale"
166
+ [name]="objectRef.nativeElement.name + '-instanced-collider-' + $index"
167
+ [options]="childColliderOption.colliderOptions"
168
+ />
169
+ }
170
+ </ngt-object3D>
171
+ }
172
+ `,
173
+ schemas: [CUSTOM_ELEMENTS_SCHEMA],
174
+ changeDetection: ChangeDetectionStrategy.OnPush,
175
+ host: {
176
+ '[position]': 'position()',
177
+ '[rotation]': 'rotation()',
178
+ '[scale]': 'scale()',
179
+ '[quaternion]': 'quaternion()',
180
+ '[userData]': 'userData()',
181
+ },
182
+ imports: [NgtrRigidBody, NgtrRigidBody, NgtrAnyCollider],
183
+ }]
184
+ }], ctorParameters: () => [] });
185
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5zdGFuY2VkLXJpZ2lkLWJvZGllcy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL2xpYnMvcmFwaWVyL3NyYy9saWIvaW5zdGFuY2VkLXJpZ2lkLWJvZGllcy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQ04sdUJBQXVCLEVBQ3ZCLFNBQVMsRUFDVCxRQUFRLEVBQ1Isc0JBQXNCLEVBQ3RCLE1BQU0sRUFDTixVQUFVLEVBQ1YsTUFBTSxFQUNOLEtBQUssRUFDTCxTQUFTLEVBQ1QsU0FBUyxFQUNULFlBQVksR0FDWixNQUFNLGVBQWUsQ0FBQztBQUN2QixPQUFPLEVBQUUsTUFBTSxFQUFFLGFBQWEsRUFBb0QsSUFBSSxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBQzlHLE9BQU8sRUFBRSxXQUFXLEVBQUUsTUFBTSwwQkFBMEIsQ0FBQztBQUN2RCxPQUFPLEVBQUUsZ0JBQWdCLEVBQWlCLFFBQVEsRUFBRSxNQUFNLE9BQU8sQ0FBQztBQUNsRSxPQUFPLEVBQUUsV0FBVyxFQUFFLE1BQU0sV0FBVyxDQUFDO0FBQ3hDLE9BQU8sRUFBRSxlQUFlLEVBQUUsYUFBYSxFQUFFLHVCQUF1QixFQUFFLE1BQU0sY0FBYyxDQUFDO0FBRXZGLE9BQU8sRUFBRSxxQkFBcUIsRUFBRSxNQUFNLFNBQVMsQ0FBQzs7QUFhaEQsTUFBTSxjQUFjLEdBQXlCLHVCQUF1QixDQUFDO0FBaURyRSxNQUFNLE9BQU8sd0JBQXdCO0lBQ3BDLFFBQVEsR0FBRyxLQUFLLENBQXlCLENBQUMsQ0FBQyxFQUFFLENBQUMsRUFBRSxDQUFDLENBQUMsQ0FBQyxDQUFDO0lBQ3BELFFBQVEsR0FBRyxLQUFLLENBQXVCLENBQUMsQ0FBQyxFQUFFLENBQUMsRUFBRSxDQUFDLENBQUMsQ0FBQyxDQUFDO0lBQ2xELEtBQUssR0FBRyxLQUFLLENBQXlCLENBQUMsQ0FBQyxFQUFFLENBQUMsRUFBRSxDQUFDLENBQUMsQ0FBQyxDQUFDO0lBQ2pELFVBQVUs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O1FBQ25DLE1BQU0sYUFBYSxHQUFHLElBQUksQ0FBQyxhQUFhLEVBQUUsQ0FBQztRQUMzQyxJQUFJLENBQUMsYUFBYTtZQUFFLE9BQU87UUFDM0IsYUFBYSxDQUFDLGNBQWMsQ0FBQyxRQUFRLENBQUMsZ0JBQWdCLENBQUMsQ0FBQztJQUN6RCxDQUFDO3VHQW5HVyx3QkFBd0I7MkZBQXhCLHdCQUF3QixxekNBZ0JQLGFBQWEsb0dBM0RoQzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztFQStCVCw0REFVUyxhQUFhLDBUQUFpQixlQUFlOzsyRkFFM0Msd0JBQXdCO2tCQS9DcEMsU0FBUzttQkFBQztvQkFDVixRQUFRLEVBQUUsd0NBQXdDO29CQUNsRCxRQUFRLEVBQUUsc0JBQXNCO29CQUNoQyxVQUFVLEVBQUUsSUFBSTtvQkFDaEIsUUFBUSxFQUFFOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0VBK0JUO29CQUNELE9BQU8sRUFBRSxDQUFDLHNCQUFzQixDQUFDO29CQUNqQyxlQUFlLEVBQUUsdUJBQXVCLENBQUMsTUFBTTtvQkFDL0MsSUFBSSxFQUFFO3dCQUNMLFlBQVksRUFBRSxZQUFZO3dCQUMxQixZQUFZLEVBQUUsWUFBWTt3QkFDMUIsU0FBUyxFQUFFLFNBQVM7d0JBQ3BCLGNBQWMsRUFBRSxjQUFjO3dCQUM5QixZQUFZLEVBQUUsWUFBWTtxQkFDMUI7b0JBQ0QsT0FBTyxFQUFFLENBQUMsYUFBYSxFQUFFLGFBQWEsRUFBRSxlQUFlLENBQUM7aUJBQ3hEIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHtcblx0Q2hhbmdlRGV0ZWN0aW9uU3RyYXRlZ3ksXG5cdENvbXBvbmVudCxcblx0Y29tcHV0ZWQsXG5cdENVU1RPTV9FTEVNRU5UU19TQ0hFTUEsXG5cdGVmZmVjdCxcblx0RWxlbWVudFJlZixcblx0aW5qZWN0LFxuXHRpbnB1dCxcblx0dW50cmFja2VkLFxuXHR2aWV3Q2hpbGQsXG5cdHZpZXdDaGlsZHJlbixcbn0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBleHRlbmQsIGdldExvY2FsU3RhdGUsIE5ndEV1bGVyLCBOZ3RPYmplY3QzRCwgTmd0UXVhdGVybmlvbiwgTmd0VmVjdG9yMywgcGljayB9IGZyb20gJ2FuZ3VsYXItdGhyZWUnO1xuaW1wb3J0IHsgbWVyZ2VJbnB1dHMgfSBmcm9tICduZ3h0ZW5zaW9uL2luamVjdC1pbnB1dHMnO1xuaW1wb3J0IHsgRHluYW1pY0RyYXdVc2FnZSwgSW5zdGFuY2VkTWVzaCwgT2JqZWN0M0QgfSBmcm9tICd0aHJlZSc7XG5pbXBvcnQgeyBOZ3RyUGh5c2ljcyB9IGZyb20gJy4vcGh5c2ljcyc7XG5pbXBvcnQgeyBOZ3RyQW55Q29sbGlkZXIsIE5ndHJSaWdpZEJvZHksIHJpZ2lkQm9keURlZmF1bHRPcHRpb25zIH0gZnJvbSAnLi9yaWdpZC1ib2R5JztcbmltcG9ydCB7IE5ndHJSaWdpZEJvZHlPcHRpb25zLCBOZ3RyUmlnaWRCb2R5U3RhdGUsIE5ndHJSaWdpZEJvZHlUeXBlIH0gZnJvbSAnLi90eXBlcyc7XG5pbXBvcnQgeyBjcmVhdGVDb2xsaWRlck9wdGlvbnMgfSBmcm9tICcuL3V0aWxzJztcblxuZXhwb3J0IGludGVyZmFjZSBOZ3RySW5zdGFuY2VkUmlnaWRCb2R5T3B0aW9ucyB7XG5cdGtleTogc3RyaW5nIHwgbnVtYmVyO1xuXHR0eXBlPzogTmd0clJpZ2lkQm9keVR5cGU7XG5cdHBvc2l0aW9uPzogTmd0VmVjdG9yMztcblx0cm90YXRpb24/OiBOZ3RFdWxlcjtcblx0c2NhbGU/OiBOZ3RWZWN0b3IzO1xuXHRxdWF0ZXJuaW9uPzogTmd0UXVhdGVybmlvbjtcblx0dXNlckRhdGE/OiBOZ3RPYmplY3QzRFsndXNlckRhdGEnXTtcblx0b3B0aW9ucz86IE5ndHJSaWdpZEJvZHlPcHRpb25zO1xufVxuXG5jb25zdCBkZWZhdWx0T3B0aW9uczogTmd0clJpZ2lkQm9keU9wdGlvbnMgPSByaWdpZEJvZHlEZWZhdWx0T3B0aW9ucztcblxuQENvbXBvbmVudCh7XG5cdHNlbGVjdG9yOiAnbmd0LW9iamVjdDNEW25ndHJJbnN0YW5jZWRSaWdpZEJvZGllc10nLFxuXHRleHBvcnRBczogJ2luc3RhbmNlZFJpZ2lkQm9kaWVzJyxcblx0c3RhbmRhbG9uZTogdHJ1ZSxcblx0dGVtcGxhdGU6IGBcblx0XHQ8bmd0LW9iamVjdDNEICNpbnN0YW5jZVdyYXBwZXI+XG5cdFx0XHQ8bmctY29udGVudCAvPlxuXHRcdDwvbmd0LW9iamVjdDNEPlxuXG5cdFx0QGZvciAoaW5zdGFuY2Ugb2YgaW5zdGFuY2VzT3B0aW9ucygpOyB0cmFjayBpbnN0YW5jZS5rZXkpIHtcblx0XHRcdDxuZ3Qtb2JqZWN0M0Rcblx0XHRcdFx0W25ndHJSaWdpZEJvZHldPVwiaW5zdGFuY2UudHlwZVwiXG5cdFx0XHRcdFtvcHRpb25zXT1cImluc3RhbmNlLm9wdGlvbnNcIlxuXHRcdFx0XHRbcG9zaXRpb25dPVwiaW5zdGFuY2UucG9zaXRpb25cIlxuXHRcdFx0XHRbcm90YXRpb25dPVwiaW5zdGFuY2Uucm90YXRpb25cIlxuXHRcdFx0XHRbc2NhbGVdPVwiaW5zdGFuY2Uuc2NhbGVcIlxuXHRcdFx0XHRbcXVhdGVybmlvbl09XCJpbnN0YW5jZS5xdWF0ZXJuaW9uXCJcblx0XHRcdFx0W3VzZXJEYXRhXT1cImluc3RhbmNlLnVzZXJEYXRhXCJcblx0XHRcdFx0W25hbWVdPVwiaW5zdGFuY2Uua2V5ICsgJy1pbnN0YW5jZWQtcmlnaWQtYm9keS0nICsgJGluZGV4XCJcblx0XHRcdD5cblx0XHRcdFx0PG5nLWNvbnRlbnQgc2VsZWN0PVwiW2RhdGEtY29sbGlkZXJzXVwiIC8+XG5cblx0XHRcdFx0QGZvciAoY2hpbGRDb2xsaWRlck9wdGlvbiBvZiBjaGlsZENvbGxpZGVyT3B0aW9ucygpOyB0cmFjayAkaW5kZXgpIHtcblx0XHRcdFx0XHQ8bmd0LW9iamVjdDNEXG5cdFx0XHRcdFx0XHRbbmd0ckNvbGxpZGVyXT1cImNoaWxkQ29sbGlkZXJPcHRpb24uc2hhcGVcIlxuXHRcdFx0XHRcdFx0W2FyZ3NdPVwiY2hpbGRDb2xsaWRlck9wdGlvbi5hcmdzXCJcblx0XHRcdFx0XHRcdFtwb3NpdGlvbl09XCJjaGlsZENvbGxpZGVyT3B0aW9uLnBvc2l0aW9uXCJcblx0XHRcdFx0XHRcdFtyb3RhdGlvbl09XCJjaGlsZENvbGxpZGVyT3B0aW9uLnJvdGF0aW9uXCJcblx0XHRcdFx0XHRcdFtzY2FsZV09XCJjaGlsZENvbGxpZGVyT3B0aW9uLnNjYWxlXCJcblx0XHRcdFx0XHRcdFtuYW1lXT1cIm9iamVjdFJlZi5uYXRpdmVFbGVtZW50Lm5hbWUgKyAnLWluc3RhbmNlZC1jb2xsaWRlci0nICsgJGluZGV4XCJcblx0XHRcdFx0XHRcdFtvcHRpb25zXT1cImNoaWxkQ29sbGlkZXJPcHRpb24uY29sbGlkZXJPcHRpb25zXCJcblx0XHRcdFx0XHQvPlxuXHRcdFx0XHR9XG5cdFx0XHQ8L25ndC1vYmplY3QzRD5cblx0XHR9XG5cdGAsXG5cdHNjaGVtYXM6IFtDVVNUT01fRUxFTUVOVFNfU0NIRU1BXSxcblx0Y2hhbmdlRGV0ZWN0aW9uOiBDaGFuZ2VEZXRlY3Rpb25TdHJhdGVneS5PblB1c2gsXG5cdGhvc3Q6IHtcblx0XHQnW3Bvc2l0aW9uXSc6ICdwb3NpdGlvbigpJyxcblx0XHQnW3JvdGF0aW9uXSc6ICdyb3RhdGlvbigpJyxcblx0XHQnW3NjYWxlXSc6ICdzY2FsZSgpJyxcblx0XHQnW3F1YXRlcm5pb25dJzogJ3F1YXRlcm5pb24oKScsXG5cdFx0J1t1c2VyRGF0YV0nOiAndXNlckRhdGEoKScsXG5cdH0sXG5cdGltcG9ydHM6IFtOZ3RyUmlnaWRCb2R5LCBOZ3RyUmlnaWRCb2R5LCBOZ3RyQW55Q29sbGlkZXJdLFxufSlcbmV4cG9ydCBjbGFzcyBOZ3RySW5zdGFuY2VkUmlnaWRCb2RpZXMge1xuXHRwb3NpdGlvbiA9IGlucHV0PE5ndFZlY3RvcjMgfCB1bmRlZmluZWQ+KFswLCAwLCAwXSk7XG5cdHJvdGF0aW9uID0gaW5wdXQ8Tmd0RXVsZXIgfCB1bmRlZmluZWQ+KFswLCAwLCAwXSk7XG5cdHNjYWxlID0gaW5wdXQ8Tmd0VmVjdG9yMyB8IHVuZGVmaW5lZD4oWzEsIDEsIDFdKTtcblx0cXVhdGVybmlvbiA9IGlucHV0PE5ndFF1YXRlcm5pb24gfCB1bmRlZmluZWQ+KFswLCAwLCAwLCAxXSk7XG5cdHVzZXJEYXRhID0gaW5wdXQ8Tmd0T2JqZWN0M0RbJ3VzZXJEYXRhJ10gfCB1bmRlZmluZWQ+KHt9KTtcblx0aW5zdGFuY2VzID0gaW5wdXQoW10sIHtcblx0XHRhbGlhczogJ25ndHJJbnN0YW5jZWRSaWdpZEJvZGllcycsXG5cdFx0dHJhbnNmb3JtOiAodmFsdWU6IEFycmF5PE5ndHJJbnN0YW5jZWRSaWdpZEJvZHlPcHRpb25zPiB8ICcnKSA9PiB7XG5cdFx0XHRpZiAodmFsdWUgPT09ICcnKSByZXR1cm4gW107XG5cdFx0XHRyZXR1cm4gdmFsdWU7XG5cdFx0fSxcblx0fSk7XG5cdG9wdGlvbnMgPSBpbnB1dChkZWZhdWx0T3B0aW9ucywgeyB0cmFuc2Zvcm06IG1lcmdlSW5wdXRzKGRlZmF1bHRPcHRpb25zKSB9KTtcblxuXHRpbnN0YW5jZVdyYXBwZXJSZWYgPSB2aWV3Q2hpbGQucmVxdWlyZWQ8RWxlbWVudFJlZjxPYmplY3QzRD4+KCdpbnN0YW5jZVdyYXBwZXInKTtcblx0cmlnaWRCb2R5UmVmcyA9IHZpZXdDaGlsZHJlbihOZ3RyUmlnaWRCb2R5KTtcblxuXHRwcml2YXRlIHBoeXNpY3MgPSBpbmplY3QoTmd0clBoeXNpY3MpO1xuXHRvYmplY3RSZWYgPSBpbmplY3Q8RWxlbWVudFJlZjxPYmplY3QzRD4+KEVsZW1lbnRSZWYpO1xuXG5cdHByaXZhdGUgY29sbGlkZXJzID0gcGljayh0aGlzLm9wdGlvbnMsICdjb2xsaWRlcnMnKTtcblxuXHRwcml2YXRlIGluc3RhbmNlZE1lc2ggPSBjb21wdXRlZCgoKSA9PiB7XG5cdFx0Y29uc3QgaW5zdGFuY2VXcmFwcGVyID0gdGhpcy5pbnN0YW5jZVdyYXBwZXJSZWYoKS5uYXRpdmVFbGVtZW50O1xuXHRcdGlmICghaW5zdGFuY2VXcmFwcGVyKSByZXR1cm4gbnVsbDtcblxuXHRcdGNvbnN0IGxvY2FsU3RhdGUgPSBnZXRMb2NhbFN0YXRlKGluc3RhbmNlV3JhcHBlcik7XG5cdFx0aWYgKCFsb2NhbFN0YXRlKSByZXR1cm4gbnVsbDtcblxuXHRcdC8vIHRyYWNrIG9iamVjdCdzIGNoaWxkcmVuXG5cdFx0bG9jYWxTdGF0ZS5vYmplY3RzKCk7XG5cdFx0Y29uc3QgZmlyc3RDaGlsZCA9IGluc3RhbmNlV3JhcHBlci5jaGlsZHJlblswXTtcblx0XHRpZiAoIWZpcnN0Q2hpbGQgfHwgIShmaXJzdENoaWxkIGFzIEluc3RhbmNlZE1lc2gpLmlzSW5zdGFuY2VkTWVzaCkgcmV0dXJuIG51bGw7XG5cblx0XHRyZXR1cm4gZmlyc3RDaGlsZCBhcyBJbnN0YW5jZWRNZXNoO1xuXHR9KTtcblxuXHRwcm90ZWN0ZWQgaW5zdGFuY2VzT3B0aW9ucyA9IGNvbXB1dGVkKCgpID0+IHtcblx0XHRjb25zdCBbaW5zdGFuY2VzLCBvcHRpb25zLCBpbnN0YW5jZWRNZXNoXSA9IFt0aGlzLmluc3RhbmNlcygpLCB1bnRyYWNrZWQodGhpcy5vcHRpb25zKSwgdGhpcy5pbnN0YW5jZWRNZXNoKCldO1xuXHRcdGlmICghaW5zdGFuY2VkTWVzaCkgcmV0dXJuIFtdO1xuXHRcdHJldHVybiBpbnN0YW5jZXMubWFwKFxuXHRcdFx0KGluc3RhbmNlLCBpbmRleCkgPT5cblx0XHRcdFx0KHtcblx0XHRcdFx0XHQuLi5pbnN0YW5jZSxcblx0XHRcdFx0XHRvcHRpb25zOiB7XG5cdFx0XHRcdFx0XHQuLi5vcHRpb25zLFxuXHRcdFx0XHRcdFx0Li4uKGluc3RhbmNlLm9wdGlvbnMgfHwge30pLFxuXHRcdFx0XHRcdFx0dHJhbnNmb3JtU3RhdGU6IChzdGF0ZSkgPT4ge1xuXHRcdFx0XHRcdFx0XHRyZXR1cm4ge1xuXHRcdFx0XHRcdFx0XHRcdC4uLnN0YXRlLFxuXHRcdFx0XHRcdFx0XHRcdGdldE1hdHJpeDogKG1hdHJpeCkgPT4ge1xuXHRcdFx0XHRcdFx0XHRcdFx0aW5zdGFuY2VkTWVzaC5nZXRNYXRyaXhBdChpbmRleCwgbWF0cml4KTtcblx0XHRcdFx0XHRcdFx0XHRcdHJldHVybiBtYXRyaXg7XG5cdFx0XHRcdFx0XHRcdFx0fSxcblx0XHRcdFx0XHRcdFx0XHRzZXRNYXRyaXg6IChtYXRyaXgpID0+IHtcblx0XHRcdFx0XHRcdFx0XHRcdGluc3RhbmNlZE1lc2guc2V0TWF0cml4QXQoaW5kZXgsIG1hdHJpeCk7XG5cdFx0XHRcdFx0XHRcdFx0XHRpbnN0YW5jZWRNZXNoLmluc3RhbmNlTWF0cml4Lm5lZWRzVXBkYXRlID0gdHJ1ZTtcblx0XHRcdFx0XHRcdFx0XHR9LFxuXHRcdFx0XHRcdFx0XHRcdG1lc2hUeXBlOiAnaW5zdGFuY2VkTWVzaCcsXG5cdFx0XHRcdFx0XHRcdH0gYXMgTmd0clJpZ2lkQm9keVN0YXRlO1xuXHRcdFx0XHRcdFx0fSxcblx0XHRcdFx0XHR9LFxuXHRcdFx0XHRcdGtleTogYCR7aW5zdGFuY2Uua2V5fS0ke2luZGV4fWAgKyBgJHtpbnN0YW5jZWRNZXNoPy51dWlkIHx8ICcnfWAsXG5cdFx0XHRcdH0pIGFzIE9taXQ8Tmd0ckluc3RhbmNlZFJpZ2lkQm9keU9wdGlvbnMsICdvcHRpb25zJz4gJiB7IG9wdGlvbnM6IFBhcnRpYWw8Tmd0clJpZ2lkQm9keU9wdGlvbnM+IH0sXG5cdFx0KTtcblx0fSk7XG5cblx0cHJvdGVjdGVkIGNoaWxkQ29sbGlkZXJPcHRpb25zID0gY29tcHV0ZWQoKCkgPT4ge1xuXHRcdGNvbnN0IGNvbGxpZGVycyA9IHRoaXMuY29sbGlkZXJzKCk7XG5cdFx0Ly8gaWYgc2VsZiBjb2xsaWRlcnMgaXMgZmFsc2UgZXhwbGljaXRseSwgZGlzYWJsZSBhdXRvIGNvbGxpZGVycyBmb3IgdGhpcyBvYmplY3QgZW50aXJlbHkuXG5cdFx0aWYgKGNvbGxpZGVycyA9PT0gZmFsc2UpIHJldHVybiBbXTtcblxuXHRcdGNvbnN0IHBoeXNpY3NDb2xsaWRlcnMgPSB0aGlzLnBoeXNpY3MuY29sbGlkZXJzKCk7XG5cdFx0Ly8gaWYgcGh5c2ljcyBjb2xsaWRlcnMgaXMgZmFsc2UgZXhwbGljaXRseSwgZGlzYWJsZSBhdXRvIGNvbGxpZGVycyBmb3IgdGhpcyBvYmplY3QgZW50aXJlbHkuXG5cdFx0aWYgKHBoeXNpY3NDb2xsaWRlcnMgPT09IGZhbHNlKSByZXR1cm4gW107XG5cblx0XHRjb25zdCBvcHRpb25zID0gdGhpcy5vcHRpb25zKCk7XG5cdFx0Ly8gaWYgY29sbGlkZXJzIG9uIG9iamVjdCBpcyBub3Qgc2V0LCB1c2UgcGh5c2ljcyBjb2xsaWRlcnNcblx0XHRpZiAoIW9wdGlvbnMuY29sbGlkZXJzKSBvcHRpb25zLmNvbGxpZGVycyA9IHBoeXNpY3NDb2xsaWRlcnM7XG5cblx0XHRjb25zdCBvYmplY3RMb2NhbFN0YXRlID0gZ2V0TG9jYWxTdGF0ZSh0aGlzLm9iamVjdFJlZi5uYXRpdmVFbGVtZW50KTtcblx0XHQvLyB0cmFjayBvYmplY3QncyBjaGlsZHJlblxuXHRcdG9iamVjdExvY2FsU3RhdGU/Lm5vbk9iamVjdHMoKTtcblxuXHRcdHJldHVybiBjcmVhdGVDb2xsaWRlck9wdGlvbnModGhpcy5vYmplY3RSZWYubmF0aXZlRWxlbWVudCwgb3B0aW9ucyk7XG5cdH0pO1xuXG5cdGNvbnN0cnVjdG9yKCkge1xuXHRcdGV4dGVuZCh7IE9iamVjdDNEIH0pO1xuXHRcdGVmZmVjdCgoKSA9PiB7XG5cdFx0XHR0aGlzLnNldEluc3RhbmNlZE1lc2hNYXRyaXhFZmZlY3QoKTtcblx0XHR9KTtcblx0fVxuXG5cdHByaXZhdGUgc2V0SW5zdGFuY2VkTWVzaE1hdHJpeEVmZmVjdCgpIHtcblx0XHRjb25zdCBpbnN0YW5jZWRNZXNoID0gdGhpcy5pbnN0YW5jZWRNZXNoKCk7XG5cdFx0aWYgKCFpbnN0YW5jZWRNZXNoKSByZXR1cm47XG5cdFx0aW5zdGFuY2VkTWVzaC5pbnN0YW5jZU1hdHJpeC5zZXRVc2FnZShEeW5hbWljRHJhd1VzYWdlKTtcblx0fVxufVxuIl19
@@ -0,0 +1,111 @@
1
+ import { computed, effect, inject } from '@angular/core';
2
+ import { resolveRef } from 'angular-three';
3
+ import { assertInjector } from 'ngxtension/assert-injector';
4
+ import { NgtrPhysics } from './physics';
5
+ import { quaternionToRapierQuaternion, vector3ToRapierVector } from './utils';
6
+ function injectImpulseJoint(bodyA, bodyB, { injector, data }) {
7
+ return assertInjector(injectImpulseJoint, injector, () => {
8
+ const physics = inject(NgtrPhysics);
9
+ const newJoint = computed(() => {
10
+ const worldSingleton = physics.worldSingleton();
11
+ if (!worldSingleton)
12
+ return null;
13
+ const a = typeof bodyA === 'function' ? resolveRef(bodyA()) : resolveRef(bodyA);
14
+ const b = typeof bodyB === 'function' ? resolveRef(bodyB()) : resolveRef(bodyB);
15
+ if (!a || !b)
16
+ return null;
17
+ const jointData = typeof data === 'function' ? data() : data;
18
+ if (!jointData)
19
+ return null;
20
+ return worldSingleton.proxy.createImpulseJoint(jointData, a, b, true);
21
+ });
22
+ effect((onCleanup) => {
23
+ const worldSingleton = physics.worldSingleton();
24
+ if (!worldSingleton)
25
+ return;
26
+ const joint = newJoint();
27
+ if (!joint)
28
+ return;
29
+ onCleanup(() => {
30
+ if (worldSingleton.proxy.getImpulseJoint(joint.handle)) {
31
+ worldSingleton.proxy.removeImpulseJoint(joint, true);
32
+ }
33
+ });
34
+ });
35
+ return newJoint;
36
+ });
37
+ }
38
+ function createJoint(jointDataFn) {
39
+ return function _injectJoint(bodyA, bodyB, { injector, data }) {
40
+ return assertInjector(_injectJoint, injector, () => {
41
+ const physics = inject(NgtrPhysics);
42
+ const jointData = computed(() => {
43
+ const rapier = physics.rapier();
44
+ if (!rapier)
45
+ return null;
46
+ return jointDataFn(rapier, data);
47
+ });
48
+ return injectImpulseJoint(bodyA, bodyB, { injector, data: jointData });
49
+ });
50
+ };
51
+ }
52
+ /**
53
+ * A fixed joint ensures that two rigid-bodies don't move relative to each other.
54
+ * Fixed joints are characterized by one local frame (represented by an isometry) on each rigid-body.
55
+ * The fixed-joint makes these frames coincide in world-space.
56
+ *
57
+ * @category Hooks - Joints
58
+ */
59
+ export const injectFixedJoint = createJoint((rapier, data) => rapier.JointData.fixed(vector3ToRapierVector(data.body1Anchor), quaternionToRapierQuaternion(data.body1LocalFrame), vector3ToRapierVector(data.body2Anchor), quaternionToRapierQuaternion(data.body2LocalFrame)));
60
+ /**
61
+ * The spherical joint ensures that two points on the local-spaces of two rigid-bodies always coincide (it prevents any relative
62
+ * translational motion at this points). This is typically used to simulate ragdolls arms, pendulums, etc.
63
+ * They are characterized by one local anchor on each rigid-body. Each anchor represents the location of the
64
+ * points that need to coincide on the local-space of each rigid-body.
65
+ *
66
+ * @category Hooks - Joints
67
+ */
68
+ export const injectSphericalJoint = createJoint((rapier, data) => rapier.JointData.spherical(vector3ToRapierVector(data.body1Anchor), vector3ToRapierVector(data.body2Anchor)));
69
+ /**
70
+ * The revolute joint prevents any relative movement between two rigid-bodies, except for relative
71
+ * rotations along one axis. This is typically used to simulate wheels, fans, etc.
72
+ * They are characterized by one local anchor as well as one local axis on each rigid-body.
73
+ *
74
+ * @category Hooks - Joints
75
+ */
76
+ export const injectRevoluteJoint = createJoint((rapier, data) => {
77
+ const jointData = rapier.JointData.revolute(vector3ToRapierVector(data.body1Anchor), vector3ToRapierVector(data.body2Anchor), vector3ToRapierVector(data.axis));
78
+ if (data.limits) {
79
+ jointData.limitsEnabled = true;
80
+ jointData.limits = data.limits;
81
+ }
82
+ return jointData;
83
+ });
84
+ /**
85
+ * The prismatic joint prevents any relative movement between two rigid-bodies, except for relative translations along one axis.
86
+ * It is characterized by one local anchor as well as one local axis on each rigid-body. In 3D, an optional
87
+ * local tangent axis can be specified for each rigid-body.
88
+ *
89
+ * @category Hooks - Joints
90
+ */
91
+ export const injectPrismaticJoint = createJoint((rapier, data) => {
92
+ const jointData = rapier.JointData.prismatic(vector3ToRapierVector(data.body1Anchor), vector3ToRapierVector(data.body2Anchor), vector3ToRapierVector(data.axis));
93
+ if (data.limits) {
94
+ jointData.limitsEnabled = true;
95
+ jointData.limits = data.limits;
96
+ }
97
+ return jointData;
98
+ });
99
+ /**
100
+ * The rope joint limits the max distance between two bodies.
101
+ * @category Hooks - Joints
102
+ */
103
+ export const injectRopeJoint = createJoint((rapier, data) => rapier.JointData.rope(data.length, vector3ToRapierVector(data.body1Anchor), vector3ToRapierVector(data.body2Anchor)));
104
+ /**
105
+ * The spring joint applies a force proportional to the distance between two objects.
106
+ * @category Hooks - Joints
107
+ */
108
+ export const injectSpringJoint = createJoint((rapier, data) => {
109
+ return rapier.JointData.spring(data.restLength, data.stiffness, data.damping, vector3ToRapierVector(data.body1Anchor), vector3ToRapierVector(data.body2Anchor));
110
+ });
111
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiam9pbnRzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vbGlicy9yYXBpZXIvc3JjL2xpYi9qb2ludHMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFFBQVEsRUFBRSxNQUFNLEVBQWMsTUFBTSxFQUFZLE1BQU0sZUFBZSxDQUFDO0FBWS9FLE9BQU8sRUFBRSxVQUFVLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFDM0MsT0FBTyxFQUFFLGNBQWMsRUFBRSxNQUFNLDRCQUE0QixDQUFDO0FBQzVELE9BQU8sRUFBRSxXQUFXLEVBQUUsTUFBTSxXQUFXLENBQUM7QUFTeEMsT0FBTyxFQUFFLDRCQUE0QixFQUFFLHFCQUFxQixFQUFFLE1BQU0sU0FBUyxDQUFDO0FBRTlFLFNBQVMsa0JBQWtCLENBQzFCLEtBQXVHLEVBQ3ZHLEtBQXVHLEVBQ3ZHLEVBQUUsUUFBUSxFQUFFLElBQUksRUFBdUU7SUFFdkYsT0FBTyxjQUFjLENBQUMsa0JBQWtCLEVBQUUsUUFBUSxFQUFFLEdBQUcsRUFBRTtRQUN4RCxNQUFNLE9BQU8sR0FBRyxNQUFNLENBQUMsV0FBVyxDQUFDLENBQUM7UUFFcEMsTUFBTSxRQUFRLEdBQUcsUUFBUSxDQUFtQixHQUFHLEVBQUU7WUFDaEQsTUFBTSxjQUFjLEdBQUcsT0FBTyxDQUFDLGNBQWMsRUFBRSxDQUFDO1lBQ2hELElBQUksQ0FBQyxjQUFjO2dCQUFFLE9BQU8sSUFBSSxDQUFDO1lBRWpDLE1BQU0sQ0FBQyxHQUFHLE9BQU8sS0FBSyxLQUFLLFVBQVUsQ0FBQyxDQUFDLENBQUMsVUFBVSxDQUFDLEtBQUssRUFBRSxDQUFDLENBQUMsQ0FBQyxDQUFDLFVBQVUsQ0FBQyxLQUFLLENBQUMsQ0FBQztZQUNoRixNQUFNLENBQUMsR0FBRyxPQUFPLEtBQUssS0FBSyxVQUFVLENBQUMsQ0FBQyxDQUFDLFVBQVUsQ0FBQyxLQUFLLEVBQUUsQ0FBQyxDQUFDLENBQUMsQ0FBQyxVQUFVLENBQUMs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QUMsTUFBTSxFQUFFLElBQUksRUFBRSxFQUFFLENBQ2xHLE1BQU0sQ0FBQyxTQUFTLENBQUMsSUFBSSxDQUFDLElBQUksQ0FBQyxNQUFNLEVBQUUscUJBQXFCLENBQUMsSUFBSSxDQUFDLFdBQVcsQ0FBQyxFQUFFLHFCQUFxQixDQUFDLElBQUksQ0FBQyxXQUFXLENBQUMsQ0FBQyxDQUNwSCxDQUFDO0FBRUY7OztHQUdHO0FBQ0gsTUFBTSxDQUFDLE1BQU0saUJBQWlCLEdBQUcsV0FBVyxDQUE0QyxDQUFDLE1BQU0sRUFBRSxJQUFJLEVBQUUsRUFBRTtJQUN4RyxPQUFPLE1BQU0sQ0FBQyxTQUFTLENBQUMsTUFBTSxDQUM3QixJQUFJLENBQUMsVUFBVSxFQUNmLElBQUksQ0FBQyxTQUFTLEVBQ2QsSUFBSSxDQUFDLE9BQU8sRUFDWixxQkFBcUIsQ0FBQyxJQUFJLENBQUMsV0FBVyxDQUFDLEVBQ3ZDLHFCQUFxQixDQUFDLElBQUksQ0FBQyxXQUFXLENBQUMsQ0FDdkMsQ0FBQztBQUNILENBQUMsQ0FBQyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgY29tcHV0ZWQsIGVmZmVjdCwgRWxlbWVudFJlZiwgaW5qZWN0LCBJbmplY3RvciB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHtcblx0Rml4ZWRJbXB1bHNlSm9pbnQsXG5cdEltcHVsc2VKb2ludCxcblx0Sm9pbnREYXRhLFxuXHRQcmlzbWF0aWNJbXB1bHNlSm9pbnQsXG5cdFJldm9sdXRlSW1wdWxzZUpvaW50LFxuXHRSaWdpZEJvZHksXG5cdFJvcGVJbXB1bHNlSm9pbnQsXG5cdFNwaGVyaWNhbEltcHVsc2VKb2ludCxcblx0U3ByaW5nSW1wdWxzZUpvaW50LFxufSBmcm9tICdAZGltZm9yZ2UvcmFwaWVyM2QtY29tcGF0JztcbmltcG9ydCB7IHJlc29sdmVSZWYgfSBmcm9tICdhbmd1bGFyLXRocmVlJztcbmltcG9ydCB7IGFzc2VydEluamVjdG9yIH0gZnJvbSAnbmd4dGVuc2lvbi9hc3NlcnQtaW5qZWN0b3InO1xuaW1wb3J0IHsgTmd0clBoeXNpY3MgfSBmcm9tICcuL3BoeXNpY3MnO1xuaW1wb3J0IHtcblx0Tmd0ckZpeGVkSm9pbnRQYXJhbXMsXG5cdE5ndHJQcmlzbWF0aWNKb2ludFBhcmFtcyxcblx0Tmd0clJldm9sdXRlSm9pbnRQYXJhbXMsXG5cdE5ndHJSb3BlSm9pbnRQYXJhbXMsXG5cdE5ndHJTcGhlcmljYWxKb2ludFBhcmFtcyxcblx0Tmd0clNwcmluZ0pvaW50UGFyYW1zLFxufSBmcm9tICcuL3R5cGVzJztcbmltcG9ydCB7IHF1YXRlcm5pb25Ub1JhcGllclF1YXRlcm5pb24sIHZlY3RvcjNUb1JhcGllclZlY3RvciB9IGZyb20gJy4vdXRpbHMnO1xuXG5mdW5jdGlvbiBpbmplY3RJbXB1bHNlSm9pbnQ8VEpvaW5UeXBlIGV4dGVuZHMgSW1wdWxzZUpvaW50Pihcblx0Ym9keUE6IEVsZW1lbnRSZWY8UmlnaWRCb2R5PiB8IFJpZ2lkQm9keSB8ICgoKSA9PiBFbGVtZW50UmVmPFJpZ2lkQm9keT4gfCBSaWdpZEJvZHkgfCB1bmRlZmluZWQgfCBudWxsKSxcblx0Ym9keUI6IEVsZW1lbnRSZWY8UmlnaWRCb2R5PiB8IFJpZ2lkQm9keSB8ICgoKSA9PiBFbGVtZW50UmVmPFJpZ2lkQm9keT4gfCBSaWdpZEJvZHkgfCB1bmRlZmluZWQgfCBudWxsKSxcblx0eyBpbmplY3RvciwgZGF0YSB9OiB7IGluamVjdG9yPzogSW5qZWN0b3I7IGRhdGE6IEpvaW50RGF0YSB8ICgoKSA9PiBKb2ludERhdGEgfCBudWxsKSB9LFxuKSB7XG5cdHJldHVybiBhc3NlcnRJbmplY3RvcihpbmplY3RJbXB1bHNlSm9pbnQsIGluamVjdG9yLCAoKSA9PiB7XG5cdFx0Y29uc3QgcGh5c2ljcyA9IGluamVjdChOZ3RyUGh5c2ljcyk7XG5cblx0XHRjb25zdCBuZXdKb2ludCA9IGNvbXB1dGVkPFRKb2luVHlwZSB8IG51bGw+KCgpID0+IHtcblx0XHRcdGNvbnN0IHdvcmxkU2luZ2xldG9uID0gcGh5c2ljcy53b3JsZFNpbmdsZXRvbigpO1xuXHRcdFx0aWYgKCF3b3JsZFNpbmdsZXRvbikgcmV0dXJuIG51bGw7XG5cblx0XHRcdGNvbnN0IGEgPSB0eXBlb2YgYm9keUEgPT09ICdmdW5jdGlvbicgPyByZXNvbHZlUmVmKGJvZHlBKCkpIDogcmVzb2x2ZVJlZihib2R5QSk7XG5cdFx0XHRjb25zdCBiID0gdHlwZW9mIGJvZHlCID09PSAnZnVuY3Rpb24nID8gcmVzb2x2ZVJlZihib2R5QigpKSA6IHJlc29sdmVSZWYoYm9keUIpO1xuXHRcdFx0aWYgKCFhIHx8ICFiKSByZXR1cm4gbnVsbDtcblxuXHRcdFx0Y29uc3Qgam9pbnREYXRhID0gdHlwZW9mIGRhdGEgPT09ICdmdW5jdGlvbicgPyBkYXRhKCkgOiBkYXRhO1xuXHRcdFx0aWYgKCFqb2ludERhdGEpIHJldHVybiBudWxsO1xuXG5cdFx0XHRyZXR1cm4gd29ybGRTaW5nbGV0b24ucHJveHkuY3JlYXRlSW1wdWxzZUpvaW50KGpvaW50RGF0YSwgYSwgYiwgdHJ1ZSkgYXMgVEpvaW5UeXBlO1xuXHRcdH0pO1xuXG5cdFx0ZWZmZWN0KChvbkNsZWFudXApID0+IHtcblx0XHRcdGNvbnN0IHdvcmxkU2luZ2xldG9uID0gcGh5c2ljcy53b3JsZFNpbmdsZXRvbigpO1xuXHRcdFx0aWYgKCF3b3JsZFNpbmdsZXRvbikgcmV0dXJuO1xuXG5cdFx0XHRjb25zdCBqb2ludCA9IG5ld0pvaW50KCk7XG5cdFx0XHRpZiAoIWpvaW50KSByZXR1cm47XG5cblx0XHRcdG9uQ2xlYW51cCgoKSA9PiB7XG5cdFx0XHRcdGlmICh3b3JsZFNpbmdsZXRvbi5wcm94eS5nZXRJbXB1bHNlSm9pbnQoam9pbnQuaGFuZGxlKSkge1xuXHRcdFx0XHRcdHdvcmxkU2luZ2xldG9uLnByb3h5LnJlbW92ZUltcHVsc2VKb2ludChqb2ludCwgdHJ1ZSk7XG5cdFx0XHRcdH1cblx0XHRcdH0pO1xuXHRcdH0pO1xuXG5cdFx0cmV0dXJuIG5ld0pvaW50O1xuXHR9KTtcbn1cblxuZnVuY3Rpb24gY3JlYXRlSm9pbnQ8VEpvaW50UGFyYW1zLCBUSm9pblR5cGUgZXh0ZW5kcyBJbXB1bHNlSm9pbnQ+KFxuXHRqb2ludERhdGFGbjogKHJhcGllcjogTm9uTnVsbGFibGU8UmV0dXJuVHlwZTxOZ3RyUGh5c2ljc1sncmFwaWVyJ10+PiwgZGF0YTogVEpvaW50UGFyYW1zKSA9PiBKb2ludERhdGEsXG4pIHtcblx0cmV0dXJuIGZ1bmN0aW9uIF9pbmplY3RKb2ludChcblx0XHRib2R5QTogRWxlbWVudFJlZjxSaWdpZEJvZHk+IHwgUmlnaWRCb2R5IHwgKCgpID0+IEVsZW1lbnRSZWY8UmlnaWRCb2R5PiB8IFJpZ2lkQm9keSB8IHVuZGVmaW5lZCB8IG51bGwpLFxuXHRcdGJvZHlCOiBFbGVtZW50UmVmPFJpZ2lkQm9keT4gfCBSaWdpZEJvZHkgfCAoKCkgPT4gRWxlbWVudFJlZjxSaWdpZEJvZHk+IHwgUmlnaWRCb2R5IHwgdW5kZWZpbmVkIHwgbnVsbCksXG5cdFx0eyBpbmplY3RvciwgZGF0YSB9OiB7IGluamVjdG9yPzogSW5qZWN0b3I7IGRhdGE6IFRKb2ludFBhcmFtcyB9LFxuXHQpIHtcblx0XHRyZXR1cm4gYXNzZXJ0SW5qZWN0b3IoX2luamVjdEpvaW50LCBpbmplY3RvciwgKCkgPT4ge1xuXHRcdFx0Y29uc3QgcGh5c2ljcyA9IGluamVjdChOZ3RyUGh5c2ljcyk7XG5cblx0XHRcdGNvbnN0IGpvaW50RGF0YSA9IGNvbXB1dGVkKCgpID0+IHtcblx0XHRcdFx0Y29uc3QgcmFwaWVyID0gcGh5c2ljcy5yYXBpZXIoKTtcblx0XHRcdFx0aWYgKCFyYXBpZXIpIHJldHVybiBudWxsO1xuXHRcdFx0XHRyZXR1cm4gam9pbnREYXRhRm4ocmFwaWVyLCBkYXRhKTtcblx0XHRcdH0pO1xuXG5cdFx0XHRyZXR1cm4gaW5qZWN0SW1wdWxzZUpvaW50PFRKb2luVHlwZT4oYm9keUEsIGJvZHlCLCB7IGluamVjdG9yLCBkYXRhOiBqb2ludERhdGEgfSk7XG5cdFx0fSk7XG5cdH07XG59XG5cbi8qKlxuICogQSBmaXhlZCBqb2ludCBlbnN1cmVzIHRoYXQgdHdvIHJpZ2lkLWJvZGllcyBkb24ndCBtb3ZlIHJlbGF0aXZlIHRvIGVhY2ggb3RoZXIuXG4gKiBGaXhlZCBqb2ludHMgYXJlIGNoYXJhY3Rlcml6ZWQgYnkgb25lIGxvY2FsIGZyYW1lIChyZXByZXNlbnRlZCBieSBhbiBpc29tZXRyeSkgb24gZWFjaCByaWdpZC1ib2R5LlxuICogVGhlIGZpeGVkLWpvaW50IG1ha2VzIHRoZXNlIGZyYW1lcyBjb2luY2lkZSBpbiB3b3JsZC1zcGFjZS5cbiAqXG4gKiBAY2F0ZWdvcnkgSG9va3MgLSBKb2ludHNcbiAqL1xuZXhwb3J0IGNvbnN0IGluamVjdEZpeGVkSm9pbnQgPSBjcmVhdGVKb2ludDxOZ3RyRml4ZWRKb2ludFBhcmFtcywgRml4ZWRJbXB1bHNlSm9pbnQ+KChyYXBpZXIsIGRhdGEpID0+XG5cdHJhcGllci5Kb2ludERhdGEuZml4ZWQoXG5cdFx0dmVjdG9yM1RvUmFwaWVyVmVjdG9yKGRhdGEuYm9keTFBbmNob3IpLFxuXHRcdHF1YXRlcm5pb25Ub1JhcGllclF1YXRlcm5pb24oZGF0YS5ib2R5MUxvY2FsRnJhbWUpLFxuXHRcdHZlY3RvcjNUb1JhcGllclZlY3RvcihkYXRhLmJvZHkyQW5jaG9yKSxcblx0XHRxdWF0ZXJuaW9uVG9SYXBpZXJRdWF0ZXJuaW9uKGRhdGEuYm9keTJMb2NhbEZyYW1lKSxcblx0KSxcbik7XG5cbi8qKlxuICogVGhlIHNwaGVyaWNhbCBqb2ludCBlbnN1cmVzIHRoYXQgdHdvIHBvaW50cyBvbiB0aGUgbG9jYWwtc3BhY2VzIG9mIHR3byByaWdpZC1ib2RpZXMgYWx3YXlzIGNvaW5jaWRlIChpdCBwcmV2ZW50cyBhbnkgcmVsYXRpdmVcbiAqIHRyYW5zbGF0aW9uYWwgbW90aW9uIGF0IHRoaXMgcG9pbnRzKS4gVGhpcyBpcyB0eXBpY2FsbHkgdXNlZCB0byBzaW11bGF0ZSByYWdkb2xscyBhcm1zLCBwZW5kdWx1bXMsIGV0Yy5cbiAqIFRoZXkgYXJlIGNoYXJhY3Rlcml6ZWQgYnkgb25lIGxvY2FsIGFuY2hvciBvbiBlYWNoIHJpZ2lkLWJvZHkuIEVhY2ggYW5jaG9yIHJlcHJlc2VudHMgdGhlIGxvY2F0aW9uIG9mIHRoZVxuICogcG9pbnRzIHRoYXQgbmVlZCB0byBjb2luY2lkZSBvbiB0aGUgbG9jYWwtc3BhY2Ugb2YgZWFjaCByaWdpZC1ib2R5LlxuICpcbiAqIEBjYXRlZ29yeSBIb29rcyAtIEpvaW50c1xuICovXG5leHBvcnQgY29uc3QgaW5qZWN0U3BoZXJpY2FsSm9pbnQgPSBjcmVhdGVKb2ludDxOZ3RyU3BoZXJpY2FsSm9pbnRQYXJhbXMsIFNwaGVyaWNhbEltcHVsc2VKb2ludD4oKHJhcGllciwgZGF0YSkgPT5cblx0cmFwaWVyLkpvaW50RGF0YS5zcGhlcmljYWwodmVjdG9yM1RvUmFwaWVyVmVjdG9yKGRhdGEuYm9keTFBbmNob3IpLCB2ZWN0b3IzVG9SYXBpZXJWZWN0b3IoZGF0YS5ib2R5MkFuY2hvcikpLFxuKTtcblxuLyoqXG4gKiBUaGUgcmV2b2x1dGUgam9pbnQgcHJldmVudHMgYW55IHJlbGF0aXZlIG1vdmVtZW50IGJldHdlZW4gdHdvIHJpZ2lkLWJvZGllcywgZXhjZXB0IGZvciByZWxhdGl2ZVxuICogcm90YXRpb25zIGFsb25nIG9uZSBheGlzLiBUaGlzIGlzIHR5cGljYWxseSB1c2VkIHRvIHNpbXVsYXRlIHdoZWVscywgZmFucywgZXRjLlxuICogVGhleSBhcmUgY2hhcmFjdGVyaXplZCBieSBvbmUgbG9jYWwgYW5jaG9yIGFzIHdlbGwgYXMgb25lIGxvY2FsIGF4aXMgb24gZWFjaCByaWdpZC1ib2R5LlxuICpcbiAqIEBjYXRlZ29yeSBIb29rcyAtIEpvaW50c1xuICovXG5leHBvcnQgY29uc3QgaW5qZWN0UmV2b2x1dGVKb2ludCA9IGNyZWF0ZUpvaW50PE5ndHJSZXZvbHV0ZUpvaW50UGFyYW1zLCBSZXZvbHV0ZUltcHVsc2VKb2ludD4oKHJhcGllciwgZGF0YSkgPT4ge1xuXHRjb25zdCBqb2ludERhdGEgPSByYXBpZXIuSm9pbnREYXRhLnJldm9sdXRlKFxuXHRcdHZlY3RvcjNUb1JhcGllclZlY3RvcihkYXRhLmJvZHkxQW5jaG9yKSxcblx0XHR2ZWN0b3IzVG9SYXBpZXJWZWN0b3IoZGF0YS5ib2R5MkFuY2hvciksXG5cdFx0dmVjdG9yM1RvUmFwaWVyVmVjdG9yKGRhdGEuYXhpcyksXG5cdCk7XG5cblx0aWYgKGRhdGEubGltaXRzKSB7XG5cdFx0am9pbnREYXRhLmxpbWl0c0VuYWJsZWQgPSB0cnVlO1xuXHRcdGpvaW50RGF0YS5saW1pdHMgPSBkYXRhLmxpbWl0cztcblx0fVxuXG5cdHJldHVybiBqb2ludERhdGE7XG59KTtcblxuLyoqXG4gKiBUaGUgcHJpc21hdGljIGpvaW50IHByZXZlbnRzIGFueSByZWxhdGl2ZSBtb3ZlbWVudCBiZXR3ZWVuIHR3byByaWdpZC1ib2RpZXMsIGV4Y2VwdCBmb3IgcmVsYXRpdmUgdHJhbnNsYXRpb25zIGFsb25nIG9uZSBheGlzLlxuICogSXQgaXMgY2hhcmFjdGVyaXplZCBieSBvbmUgbG9jYWwgYW5jaG9yIGFzIHdlbGwgYXMgb25lIGxvY2FsIGF4aXMgb24gZWFjaCByaWdpZC1ib2R5LiBJbiAzRCwgYW4gb3B0aW9uYWxcbiAqIGxvY2FsIHRhbmdlbnQgYXhpcyBjYW4gYmUgc3BlY2lmaWVkIGZvciBlYWNoIHJpZ2lkLWJvZHkuXG4gKlxuICogQGNhdGVnb3J5IEhvb2tzIC0gSm9pbnRzXG4gKi9cbmV4cG9ydCBjb25zdCBpbmplY3RQcmlzbWF0aWNKb2ludCA9IGNyZWF0ZUpvaW50PE5ndHJQcmlzbWF0aWNKb2ludFBhcmFtcywgUHJpc21hdGljSW1wdWxzZUpvaW50PigocmFwaWVyLCBkYXRhKSA9PiB7XG5cdGNvbnN0IGpvaW50RGF0YSA9IHJhcGllci5Kb2ludERhdGEucHJpc21hdGljKFxuXHRcdHZlY3RvcjNUb1JhcGllclZlY3RvcihkYXRhLmJvZHkxQW5jaG9yKSxcblx0XHR2ZWN0b3IzVG9SYXBpZXJWZWN0b3IoZGF0YS5ib2R5MkFuY2hvciksXG5cdFx0dmVjdG9yM1RvUmFwaWVyVmVjdG9yKGRhdGEuYXhpcyksXG5cdCk7XG5cblx0aWYgKGRhdGEubGltaXRzKSB7XG5cdFx0am9pbnREYXRhLmxpbWl0c0VuYWJsZWQgPSB0cnVlO1xuXHRcdGpvaW50RGF0YS5saW1pdHMgPSBkYXRhLmxpbWl0cztcblx0fVxuXG5cdHJldHVybiBqb2ludERhdGE7XG59KTtcblxuLyoqXG4gKiBUaGUgcm9wZSBqb2ludCBsaW1pdHMgdGhlIG1heCBkaXN0YW5jZSBiZXR3ZWVuIHR3byBib2RpZXMuXG4gKiBAY2F0ZWdvcnkgSG9va3MgLSBKb2ludHNcbiAqL1xuZXhwb3J0IGNvbnN0IGluamVjdFJvcGVKb2ludCA9IGNyZWF0ZUpvaW50PE5ndHJSb3BlSm9pbnRQYXJhbXMsIFJvcGVJbXB1bHNlSm9pbnQ+KChyYXBpZXIsIGRhdGEpID0+XG5cdHJhcGllci5Kb2ludERhdGEucm9wZShkYXRhLmxlbmd0aCwgdmVjdG9yM1RvUmFwaWVyVmVjdG9yKGRhdGEuYm9keTFBbmNob3IpLCB2ZWN0b3IzVG9SYXBpZXJWZWN0b3IoZGF0YS5ib2R5MkFuY2hvcikpLFxuKTtcblxuLyoqXG4gKiBUaGUgc3ByaW5nIGpvaW50IGFwcGxpZXMgYSBmb3JjZSBwcm9wb3J0aW9uYWwgdG8gdGhlIGRpc3RhbmNlIGJldHdlZW4gdHdvIG9iamVjdHMuXG4gKiBAY2F0ZWdvcnkgSG9va3MgLSBKb2ludHNcbiAqL1xuZXhwb3J0IGNvbnN0IGluamVjdFNwcmluZ0pvaW50ID0gY3JlYXRlSm9pbnQ8Tmd0clNwcmluZ0pvaW50UGFyYW1zLCBTcHJpbmdJbXB1bHNlSm9pbnQ+KChyYXBpZXIsIGRhdGEpID0+IHtcblx0cmV0dXJuIHJhcGllci5Kb2ludERhdGEuc3ByaW5nKFxuXHRcdGRhdGEucmVzdExlbmd0aCxcblx0XHRkYXRhLnN0aWZmbmVzcyxcblx0XHRkYXRhLmRhbXBpbmcsXG5cdFx0dmVjdG9yM1RvUmFwaWVyVmVjdG9yKGRhdGEuYm9keTFBbmNob3IpLFxuXHRcdHZlY3RvcjNUb1JhcGllclZlY3RvcihkYXRhLmJvZHkyQW5jaG9yKSxcblx0KTtcbn0pO1xuIl19
@@ -0,0 +1,69 @@
1
+ import { ChangeDetectionStrategy, Component, computed, CUSTOM_ELEMENTS_SCHEMA, ElementRef, inject, input, } from '@angular/core';
2
+ import { extend, getLocalState } from 'angular-three';
3
+ import { Object3D } from 'three';
4
+ import { NgtrPhysics } from './physics';
5
+ import { NgtrAnyCollider, NgtrRigidBody } from './rigid-body';
6
+ import { createColliderOptions } from './utils';
7
+ import * as i0 from "@angular/core";
8
+ export class NgtrMeshCollider {
9
+ colliders = input.required({ alias: 'ngtrMeshCollider' });
10
+ objectRef = inject(ElementRef);
11
+ rigidBody = inject(NgtrRigidBody);
12
+ physics = inject(NgtrPhysics);
13
+ childColliderOptions = computed(() => {
14
+ const rigidBodyOptions = this.rigidBody.options();
15
+ rigidBodyOptions.colliders = this.colliders();
16
+ const objectLocalState = getLocalState(this.objectRef.nativeElement);
17
+ // track object's children
18
+ objectLocalState?.nonObjects();
19
+ objectLocalState?.objects();
20
+ return createColliderOptions(this.objectRef.nativeElement, rigidBodyOptions, false);
21
+ });
22
+ constructor() {
23
+ extend({ Object3D });
24
+ if (!this.objectRef.nativeElement.userData) {
25
+ this.objectRef.nativeElement.userData = {};
26
+ }
27
+ this.objectRef.nativeElement.userData['ngtrRapierType'] = 'MeshCollider';
28
+ }
29
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.0", ngImport: i0, type: NgtrMeshCollider, deps: [], target: i0.ɵɵFactoryTarget.Component });
30
+ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.0", type: NgtrMeshCollider, isStandalone: true, selector: "ngt-object3D[ngtrMeshCollider]", inputs: { colliders: { classPropertyName: "colliders", publicName: "ngtrMeshCollider", isSignal: true, isRequired: true, transformFunction: null } }, ngImport: i0, template: `
31
+ <ng-content />
32
+ @for (childColliderOption of childColliderOptions(); track $index) {
33
+ <ngt-object3D
34
+ [ngtrCollider]="childColliderOption.shape"
35
+ [args]="childColliderOption.args"
36
+ [position]="childColliderOption.position"
37
+ [rotation]="childColliderOption.rotation"
38
+ [scale]="childColliderOption.scale"
39
+ [name]="objectRef.nativeElement.name + '-mesh-collider-' + $index"
40
+ [options]="childColliderOption.colliderOptions"
41
+ />
42
+ }
43
+ `, isInline: true, dependencies: [{ kind: "directive", type: NgtrAnyCollider, selector: "ngt-object3D[ngtrCollider]", inputs: ["position", "rotation", "scale", "quaternion", "userData", "name", "options", "ngtrCollider", "args"], outputs: ["ngtrColliderChange", "argsChange", "collisionEnter", "collisionExit", "intersectionEnter", "intersectionExit", "contactForce"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
44
+ }
45
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.0", ngImport: i0, type: NgtrMeshCollider, decorators: [{
46
+ type: Component,
47
+ args: [{
48
+ selector: 'ngt-object3D[ngtrMeshCollider]',
49
+ standalone: true,
50
+ template: `
51
+ <ng-content />
52
+ @for (childColliderOption of childColliderOptions(); track $index) {
53
+ <ngt-object3D
54
+ [ngtrCollider]="childColliderOption.shape"
55
+ [args]="childColliderOption.args"
56
+ [position]="childColliderOption.position"
57
+ [rotation]="childColliderOption.rotation"
58
+ [scale]="childColliderOption.scale"
59
+ [name]="objectRef.nativeElement.name + '-mesh-collider-' + $index"
60
+ [options]="childColliderOption.colliderOptions"
61
+ />
62
+ }
63
+ `,
64
+ schemas: [CUSTOM_ELEMENTS_SCHEMA],
65
+ changeDetection: ChangeDetectionStrategy.OnPush,
66
+ imports: [NgtrAnyCollider],
67
+ }]
68
+ }], ctorParameters: () => [] });
69
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibWVzaC1jb2xsaWRlci5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL2xpYnMvcmFwaWVyL3NyYy9saWIvbWVzaC1jb2xsaWRlci50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQ04sdUJBQXVCLEVBQ3ZCLFNBQVMsRUFDVCxRQUFRLEVBQ1Isc0JBQXNCLEVBQ3RCLFVBQVUsRUFDVixNQUFNLEVBQ04sS0FBSyxHQUNMLE1BQU0sZUFBZSxDQUFDO0FBQ3ZCLE9BQU8sRUFBRSxNQUFNLEVBQUUsYUFBYSxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBQ3RELE9BQU8sRUFBRSxRQUFRLEVBQUUsTUFBTSxPQUFPLENBQUM7QUFDakMsT0FBTyxFQUFFLFdBQVcsRUFBRSxNQUFNLFdBQVcsQ0FBQztBQUN4QyxPQUFPLEVBQUUsZUFBZSxFQUFFLGFBQWEsRUFBRSxNQUFNLGNBQWMsQ0FBQztBQUU5RCxPQUFPLEVBQUUscUJBQXFCLEVBQUUsTUFBTSxTQUFTLENBQUM7O0FBdUJoRCxNQUFNLE9BQU8sZ0JBQWdCO0lBQzVCLFNBQVMsR0FBRyxLQUFLLENBQUMsUUFBUSxDQUE0QixFQUFFLEtBQUssRUFBRSxrQkFBa0IsRUFBRSxDQUFDLENBQUM7SUFFckYsU0FBUyxHQUFHLE1BQU0sQ0FBdUIsVUFBVSxDQUFDLENBQUM7SUFDckQsU0FBUyxHQUFHLE1BQU0sQ0FBQyxhQUFhLENBQUMsQ0FBQztJQUNsQyxPQUFPLEdBQUcsTUFBTSxDQUFDLFdBQVcsQ0FBQyxDQUFDO0lBRXBCLG9CQUFvQixHQUFHLFFBQVEsQ0FBQyxHQUFHLEVBQUU7UUFDOUMsTUFBTSxnQkFBZ0IsR0FBRyxJQUFJLENBQUMsU0FBUyxDQUFDLE9BQU8sRUFBRSxDQUFDO1FBQ2xELGdCQUFnQixDQUFDLFNBQVMsR0FBRyxJQUFJLENBQUMsU0FBUyxFQUFFLENBQUM7UUFFOUMsTUFBTSxnQkFBZ0IsR0FBRyxhQUFhLENBQUMsSUFBSSxDQUFDLFNBQVMsQ0FBQyxhQUFhLENBQUMsQ0FBQztRQUNyRSwwQkFBMEI7UUFDMUIsZ0JBQWdCLEVBQUUsVUFBVSxFQUFFLENBQUM7UUFDL0IsZ0JBQWdCLEVBQUUsT0FBTyxFQUFFLENBQUM7UUFFNUIsT0FBTyxxQkFBcUIsQ0FBQyxJQUFJLENBQUMsU0FBUyxDQUFDLGFBQWEsRUFBRSxnQkFBZ0IsRUFBRSxLQUFLLENBQUMsQ0FBQztJQUNyRixDQUFDLENBQUMsQ0FBQztJQUVIO1FBQ0MsTUFBTSxDQUFDLEVBQUUsUUFBUSxFQUFFLENBQUMsQ0FBQztRQUNyQixJQUFJLENBQUMsSUFBSSxDQUFDLFNBQVMsQ0FBQyxhQUFhLENBQUMsUUFBUSxFQUFFLENBQUM7WUFDNUMsSUFBSSxDQUFDLFNBQVMsQ0FBQyxhQUFhLENBQUMsUUFBUSxHQUFHLEVBQUUsQ0FBQztRQUM1QyxDQUFDO1FBQ0QsSUFBSSxDQUFDLFNBQVMsQ0FBQyxhQUFhLENBQUMsUUFBUSxDQUFDLGdCQUFnQixDQUFDLEdBQUcsY0FBYyxDQUFDO0lBQzFFLENBQUM7dUdBekJXLGdCQUFnQjsyRkFBaEIsZ0JBQWdCLGdQQWxCbEI7Ozs7Ozs7Ozs7Ozs7RUFhVCw0REFHUyxlQUFlOzsyRkFFYixnQkFBZ0I7a0JBckI1QixTQUFTO21CQUFDO29CQUNWLFFBQVEsRUFBRSxnQ0FBZ0M7b0JBQzFDLFVBQVUsRUFBRSxJQUFJO29CQUNoQixRQUFRLEVBQUU7Ozs7Ozs7Ozs7Ozs7RUFhVDtvQkFDRCxPQUFPLEVBQUUsQ0FBQyxzQkFBc0IsQ0FBQztvQkFDakMsZUFBZSxFQUFFLHVCQUF1QixDQUFDLE1BQU07b0JBQy9DLE9BQU8sRUFBRSxDQUFDLGVBQWUsQ0FBQztpQkFDMUIiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQge1xuXHRDaGFuZ2VEZXRlY3Rpb25TdHJhdGVneSxcblx0Q29tcG9uZW50LFxuXHRjb21wdXRlZCxcblx0Q1VTVE9NX0VMRU1FTlRTX1NDSEVNQSxcblx0RWxlbWVudFJlZixcblx0aW5qZWN0LFxuXHRpbnB1dCxcbn0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBleHRlbmQsIGdldExvY2FsU3RhdGUgfSBmcm9tICdhbmd1bGFyLXRocmVlJztcbmltcG9ydCB7IE9iamVjdDNEIH0gZnJvbSAndGhyZWUnO1xuaW1wb3J0IHsgTmd0clBoeXNpY3MgfSBmcm9tICcuL3BoeXNpY3MnO1xuaW1wb3J0IHsgTmd0ckFueUNvbGxpZGVyLCBOZ3RyUmlnaWRCb2R5IH0gZnJvbSAnLi9yaWdpZC1ib2R5JztcbmltcG9ydCB7IE5ndHJSaWdpZEJvZHlBdXRvQ29sbGlkZXIgfSBmcm9tICcuL3R5cGVzJztcbmltcG9ydCB7IGNyZWF0ZUNvbGxpZGVyT3B0aW9ucyB9IGZyb20gJy4vdXRpbHMnO1xuXG5AQ29tcG9uZW50KHtcblx0c2VsZWN0b3I6ICduZ3Qtb2JqZWN0M0Rbbmd0ck1lc2hDb2xsaWRlcl0nLFxuXHRzdGFuZGFsb25lOiB0cnVlLFxuXHR0ZW1wbGF0ZTogYFxuXHRcdDxuZy1jb250ZW50IC8+XG5cdFx0QGZvciAoY2hpbGRDb2xsaWRlck9wdGlvbiBvZiBjaGlsZENvbGxpZGVyT3B0aW9ucygpOyB0cmFjayAkaW5kZXgpIHtcblx0XHRcdDxuZ3Qtb2JqZWN0M0Rcblx0XHRcdFx0W25ndHJDb2xsaWRlcl09XCJjaGlsZENvbGxpZGVyT3B0aW9uLnNoYXBlXCJcblx0XHRcdFx0W2FyZ3NdPVwiY2hpbGRDb2xsaWRlck9wdGlvbi5hcmdzXCJcblx0XHRcdFx0W3Bvc2l0aW9uXT1cImNoaWxkQ29sbGlkZXJPcHRpb24ucG9zaXRpb25cIlxuXHRcdFx0XHRbcm90YXRpb25dPVwiY2hpbGRDb2xsaWRlck9wdGlvbi5yb3RhdGlvblwiXG5cdFx0XHRcdFtzY2FsZV09XCJjaGlsZENvbGxpZGVyT3B0aW9uLnNjYWxlXCJcblx0XHRcdFx0W25hbWVdPVwib2JqZWN0UmVmLm5hdGl2ZUVsZW1lbnQubmFtZSArICctbWVzaC1jb2xsaWRlci0nICsgJGluZGV4XCJcblx0XHRcdFx0W29wdGlvbnNdPVwiY2hpbGRDb2xsaWRlck9wdGlvbi5jb2xsaWRlck9wdGlvbnNcIlxuXHRcdFx0Lz5cblx0XHR9XG5cdGAsXG5cdHNjaGVtYXM6IFtDVVNUT01fRUxFTUVOVFNfU0NIRU1BXSxcblx0Y2hhbmdlRGV0ZWN0aW9uOiBDaGFuZ2VEZXRlY3Rpb25TdHJhdGVneS5PblB1c2gsXG5cdGltcG9ydHM6IFtOZ3RyQW55Q29sbGlkZXJdLFxufSlcbmV4cG9ydCBjbGFzcyBOZ3RyTWVzaENvbGxpZGVyIHtcblx0Y29sbGlkZXJzID0gaW5wdXQucmVxdWlyZWQ8Tmd0clJpZ2lkQm9keUF1dG9Db2xsaWRlcj4oeyBhbGlhczogJ25ndHJNZXNoQ29sbGlkZXInIH0pO1xuXG5cdG9iamVjdFJlZiA9IGluamVjdDxFbGVtZW50UmVmPE9iamVjdDNEPj4oRWxlbWVudFJlZik7XG5cdHJpZ2lkQm9keSA9IGluamVjdChOZ3RyUmlnaWRCb2R5KTtcblx0cGh5c2ljcyA9IGluamVjdChOZ3RyUGh5c2ljcyk7XG5cblx0cHJvdGVjdGVkIGNoaWxkQ29sbGlkZXJPcHRpb25zID0gY29tcHV0ZWQoKCkgPT4ge1xuXHRcdGNvbnN0IHJpZ2lkQm9keU9wdGlvbnMgPSB0aGlzLnJpZ2lkQm9keS5vcHRpb25zKCk7XG5cdFx0cmlnaWRCb2R5T3B0aW9ucy5jb2xsaWRlcnMgPSB0aGlzLmNvbGxpZGVycygpO1xuXG5cdFx0Y29uc3Qgb2JqZWN0TG9jYWxTdGF0ZSA9IGdldExvY2FsU3RhdGUodGhpcy5vYmplY3RSZWYubmF0aXZlRWxlbWVudCk7XG5cdFx0Ly8gdHJhY2sgb2JqZWN0J3MgY2hpbGRyZW5cblx0XHRvYmplY3RMb2NhbFN0YXRlPy5ub25PYmplY3RzKCk7XG5cdFx0b2JqZWN0TG9jYWxTdGF0ZT8ub2JqZWN0cygpO1xuXG5cdFx0cmV0dXJuIGNyZWF0ZUNvbGxpZGVyT3B0aW9ucyh0aGlzLm9iamVjdFJlZi5uYXRpdmVFbGVtZW50LCByaWdpZEJvZHlPcHRpb25zLCBmYWxzZSk7XG5cdH0pO1xuXG5cdGNvbnN0cnVjdG9yKCkge1xuXHRcdGV4dGVuZCh7IE9iamVjdDNEIH0pO1xuXHRcdGlmICghdGhpcy5vYmplY3RSZWYubmF0aXZlRWxlbWVudC51c2VyRGF0YSkge1xuXHRcdFx0dGhpcy5vYmplY3RSZWYubmF0aXZlRWxlbWVudC51c2VyRGF0YSA9IHt9O1xuXHRcdH1cblx0XHR0aGlzLm9iamVjdFJlZi5uYXRpdmVFbGVtZW50LnVzZXJEYXRhWyduZ3RyUmFwaWVyVHlwZSddID0gJ01lc2hDb2xsaWRlcic7XG5cdH1cbn1cbiJdfQ==