@colijnit/configurator 1.0.17 → 1.0.19
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/app/builder.d.ts +53 -53
- package/app/components/answers/answer/answer.component.d.ts +11 -11
- package/app/components/answers/answers.component.d.ts +8 -8
- package/app/components/answers/answers.module.d.ts +2 -2
- package/app/components/selections/selections.component.d.ts +22 -22
- package/app/components/selections/selections.module.d.ts +2 -2
- package/app/components/shared/loader/loader.component.d.ts +2 -2
- package/app/components/shared/shared.module.d.ts +2 -2
- package/app/directives/visibility-observer-master.directive.d.ts +9 -9
- package/app/directives/visibility-observer.directive.d.ts +13 -13
- package/app/services/configurator.service.d.ts +22 -22
- package/app/services/image-cache.service.d.ts +10 -10
- package/app/services/locator.service.d.ts +4 -4
- package/bundles/colijnit-configurator.umd.js +2455 -2455
- package/bundles/colijnit-configurator.umd.js.map +1 -1
- package/bundles/colijnit-configurator.umd.min.js +1 -1
- package/bundles/colijnit-configurator.umd.min.js.map +1 -1
- package/colijnit-configurator.d.ts +10 -10
- package/colijnit-configurator.metadata.json +1 -1
- package/esm2015/app/builder.js +477 -477
- package/esm2015/app/components/answers/answer/answer.component.js +51 -51
- package/esm2015/app/components/answers/answers.component.js +25 -25
- package/esm2015/app/components/answers/answers.module.js +28 -28
- package/esm2015/app/components/selections/selections.component.js +85 -85
- package/esm2015/app/components/selections/selections.module.js +22 -22
- package/esm2015/app/components/shared/loader/loader.component.js +12 -12
- package/esm2015/app/components/shared/shared.module.js +20 -20
- package/esm2015/app/directives/visibility-observer-master.directive.js +50 -50
- package/esm2015/app/directives/visibility-observer.directive.js +56 -56
- package/esm2015/app/services/configurator.service.js +94 -94
- package/esm2015/app/services/image-cache.service.js +66 -66
- package/esm2015/app/services/locator.service.js +9 -9
- package/esm2015/colijnit-configurator.js +10 -10
- package/esm2015/helper/variation-helper.js +215 -215
- package/esm2015/model/material.js +10 -10
- package/esm2015/model/variation-settings.js +5 -5
- package/esm2015/model/variation.js +2 -2
- package/esm2015/public_api.js +6 -6
- package/esm2015/utils/asset.utils.js +73 -73
- package/esm2015/utils/file.utils.js +138 -138
- package/esm2015/utils/image.utils.js +51 -51
- package/esm2015/utils/object.utils.js +48 -48
- package/esm2015/utils/scene-utils.js +93 -93
- package/esm2015/utils/threed.utils.js +221 -221
- package/esm2015/utils/variation-utils.js +223 -223
- package/esm5/app/builder.js +591 -591
- package/esm5/app/components/answers/answer/answer.component.js +63 -63
- package/esm5/app/components/answers/answers.component.js +26 -26
- package/esm5/app/components/answers/answers.module.js +31 -31
- package/esm5/app/components/selections/selections.component.js +103 -103
- package/esm5/app/components/selections/selections.module.js +25 -25
- package/esm5/app/components/shared/loader/loader.component.js +15 -15
- package/esm5/app/components/shared/shared.module.js +23 -23
- package/esm5/app/directives/visibility-observer-master.directive.js +63 -63
- package/esm5/app/directives/visibility-observer.directive.js +58 -58
- package/esm5/app/services/configurator.service.js +160 -160
- package/esm5/app/services/image-cache.service.js +69 -69
- package/esm5/app/services/locator.service.js +12 -12
- package/esm5/colijnit-configurator.js +10 -10
- package/esm5/helper/variation-helper.js +267 -267
- package/esm5/model/material.js +12 -12
- package/esm5/model/variation-settings.js +7 -7
- package/esm5/model/variation.js +6 -6
- package/esm5/public_api.js +6 -6
- package/esm5/utils/asset.utils.js +105 -105
- package/esm5/utils/file.utils.js +150 -150
- package/esm5/utils/image.utils.js +55 -55
- package/esm5/utils/object.utils.js +55 -55
- package/esm5/utils/scene-utils.js +97 -97
- package/esm5/utils/threed.utils.js +278 -278
- package/esm5/utils/variation-utils.js +326 -326
- package/fesm2015/colijnit-configurator.js +1946 -1946
- package/fesm2015/colijnit-configurator.js.map +1 -1
- package/fesm5/colijnit-configurator.js +2457 -2457
- package/fesm5/colijnit-configurator.js.map +1 -1
- package/helper/variation-helper.d.ts +14 -14
- package/model/material.d.ts +17 -17
- package/model/variation-settings.d.ts +14 -14
- package/model/variation.d.ts +10 -10
- package/package.json +3 -2
- package/public_api.d.ts +6 -6
- package/utils/asset.utils.d.ts +13 -13
- package/utils/file.utils.d.ts +27 -27
- package/utils/image.utils.d.ts +8 -8
- package/utils/object.utils.d.ts +7 -7
- package/utils/scene-utils.d.ts +7 -7
- package/utils/threed.utils.d.ts +16 -16
- package/utils/variation-utils.d.ts +12 -12
|
@@ -1,49 +1,49 @@
|
|
|
1
|
-
import { Mesh } from "three";
|
|
2
|
-
export class ObjectUtils {
|
|
3
|
-
static DisposeMaterial(material) {
|
|
4
|
-
if (!material) {
|
|
5
|
-
return;
|
|
6
|
-
}
|
|
7
|
-
if (Array.isArray(material)) {
|
|
8
|
-
material.forEach(mtrl => {
|
|
9
|
-
this.materialProps
|
|
10
|
-
.filter(mapType => mtrl[mapType])
|
|
11
|
-
.forEach(mapType => mtrl[mapType].dispose());
|
|
12
|
-
mtrl.dispose();
|
|
13
|
-
});
|
|
14
|
-
}
|
|
15
|
-
else {
|
|
16
|
-
this.materialProps
|
|
17
|
-
.filter(mapType => material[mapType] && material[mapType].dispose)
|
|
18
|
-
.forEach(mapType => material[mapType].dispose());
|
|
19
|
-
if (material.dispose) {
|
|
20
|
-
material.dispose();
|
|
21
|
-
}
|
|
22
|
-
}
|
|
23
|
-
}
|
|
24
|
-
static DisposeObject(object) {
|
|
25
|
-
if (!object) {
|
|
26
|
-
return;
|
|
27
|
-
}
|
|
28
|
-
if (object.children.length) {
|
|
29
|
-
object.children.forEach(child => this.DisposeObject(child));
|
|
30
|
-
}
|
|
31
|
-
this.DisposeNode(object);
|
|
32
|
-
}
|
|
33
|
-
static DisposeNode(node) {
|
|
34
|
-
node.traverse((obj) => {
|
|
35
|
-
if (obj instanceof Mesh) {
|
|
36
|
-
if (obj.geometry) {
|
|
37
|
-
obj.geometry.dispose();
|
|
38
|
-
}
|
|
39
|
-
this.DisposeMaterial(obj.material);
|
|
40
|
-
}
|
|
41
|
-
if (typeof obj.dispose === 'function') {
|
|
42
|
-
// @ts-ignore
|
|
43
|
-
obj.dispose();
|
|
44
|
-
}
|
|
45
|
-
});
|
|
46
|
-
}
|
|
47
|
-
}
|
|
48
|
-
ObjectUtils.materialProps = ['map', 'lightMap', 'bumpMap', 'normalMap', 'specularMap', 'envMap', 'aoMap', 'roughnessMap', 'metalnessMap'];
|
|
1
|
+
import { Mesh } from "three";
|
|
2
|
+
export class ObjectUtils {
|
|
3
|
+
static DisposeMaterial(material) {
|
|
4
|
+
if (!material) {
|
|
5
|
+
return;
|
|
6
|
+
}
|
|
7
|
+
if (Array.isArray(material)) {
|
|
8
|
+
material.forEach(mtrl => {
|
|
9
|
+
this.materialProps
|
|
10
|
+
.filter(mapType => mtrl[mapType])
|
|
11
|
+
.forEach(mapType => mtrl[mapType].dispose());
|
|
12
|
+
mtrl.dispose();
|
|
13
|
+
});
|
|
14
|
+
}
|
|
15
|
+
else {
|
|
16
|
+
this.materialProps
|
|
17
|
+
.filter(mapType => material[mapType] && material[mapType].dispose)
|
|
18
|
+
.forEach(mapType => material[mapType].dispose());
|
|
19
|
+
if (material.dispose) {
|
|
20
|
+
material.dispose();
|
|
21
|
+
}
|
|
22
|
+
}
|
|
23
|
+
}
|
|
24
|
+
static DisposeObject(object) {
|
|
25
|
+
if (!object) {
|
|
26
|
+
return;
|
|
27
|
+
}
|
|
28
|
+
if (object.children.length) {
|
|
29
|
+
object.children.forEach(child => this.DisposeObject(child));
|
|
30
|
+
}
|
|
31
|
+
this.DisposeNode(object);
|
|
32
|
+
}
|
|
33
|
+
static DisposeNode(node) {
|
|
34
|
+
node.traverse((obj) => {
|
|
35
|
+
if (obj instanceof Mesh) {
|
|
36
|
+
if (obj.geometry) {
|
|
37
|
+
obj.geometry.dispose();
|
|
38
|
+
}
|
|
39
|
+
this.DisposeMaterial(obj.material);
|
|
40
|
+
}
|
|
41
|
+
if (typeof obj.dispose === 'function') {
|
|
42
|
+
// @ts-ignore
|
|
43
|
+
obj.dispose();
|
|
44
|
+
}
|
|
45
|
+
});
|
|
46
|
+
}
|
|
47
|
+
}
|
|
48
|
+
ObjectUtils.materialProps = ['map', 'lightMap', 'bumpMap', 'normalMap', 'specularMap', 'envMap', 'aoMap', 'roughnessMap', 'metalnessMap'];
|
|
49
49
|
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoib2JqZWN0LnV0aWxzLmpzIiwic291cmNlUm9vdCI6Im5nOi8vQGNvbGlqbml0L2NvbmZpZ3VyYXRvci8iLCJzb3VyY2VzIjpbInV0aWxzL29iamVjdC51dGlscy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQVcsSUFBSSxFQUFXLE1BQU0sT0FBTyxDQUFDO0FBRS9DLE1BQU0sT0FBTyxXQUFXO0lBR2IsTUFBTSxDQUFDLGVBQWUsQ0FBQyxRQUErQjtRQUN6RCxJQUFJLENBQUMsUUFBUSxFQUFFO1lBQ1gsT0FBTztTQUNWO1FBQ0QsSUFBSSxLQUFLLENBQUMsT0FBTyxDQUFDLFFBQVEsQ0FBQyxFQUFFO1lBQ3pCLFFBQVEsQ0FBQyxPQUFPLENBQUMsSUFBSSxDQUFDLEVBQUU7Z0JBQ3BCLElBQUksQ0FBQyxhQUFhO3FCQUNiLE1BQU0sQ0FBQyxPQUFPLENBQUMsRUFBRSxDQUFDLElBQUksQ0FBQyxPQUFPLENBQUMsQ0FBQztxQkFDaEMsT0FBTyxDQUFDLE9BQU8sQ0FBQyxFQUFFLENBQUMsSUFBSSxDQUFDLE9BQU8sQ0FBQyxDQUFDLE9BQU8sRUFBRSxDQUFDLENBQUM7Z0JBQ2pELElBQUksQ0FBQyxPQUFPLEVBQUUsQ0FBQztZQUNuQixDQUFDLENBQUMsQ0FBQztTQUNOO2FBQU07WUFDSCxJQUFJLENBQUMsYUFBYTtpQkFDYixNQUFNLENBQUMsT0FBTyxDQUFDLEVBQUUsQ0FBQyxRQUFRLENBQUMsT0FBTyxDQUFDLElBQUksUUFBUSxDQUFDLE9BQU8sQ0FBQyxDQUFDLE9BQU8sQ0FBQztpQkFDakUsT0FBTyxDQUFDLE9BQU8sQ0FBQyxFQUFFLENBQUMsUUFBUSxDQUFDLE9BQU8sQ0FBQyxDQUFDLE9BQU8sRUFBRSxDQUFDLENBQUM7WUFDckQsSUFBSSxRQUFRLENBQUMsT0FBTyxFQUFFO2dCQUNsQixRQUFRLENBQUMsT0FBTyxFQUFFLENBQUM7YUFDdEI7U0FDSjtJQUNMLENBQUM7SUFFTSxNQUFNLENBQUMsYUFBYSxDQUFDLE1BQXVCO1FBQy9DLElBQUksQ0FBQyxNQUFNLEVBQUU7WUFDVCxPQUFPO1NBQ1Y7UUFDRCxJQUFJLE1BQU0sQ0FBQyxRQUFRLENBQUMsTUFBTSxFQUFFO1lBQ3hCLE1BQU0sQ0FBQyxRQUFRLENBQUMsT0FBTyxDQUFDLEtBQUssQ0FBQyxFQUFFLENBQUMsSUFBSSxDQUFDLGFBQWEsQ0FBQyxLQUFLLENBQUMsQ0FBQyxDQUFDO1NBQy9EO1FBQ0QsSUFBSSxDQUFDLFdBQVcsQ0FBQyxNQUFNLENBQUMsQ0FBQztJQUM3QixDQUFDO0lBRU8sTUFBTSxDQUFDLFdBQVcsQ0FBQyxJQUFxQjtRQUM1QyxJQUFJLENBQUMsUUFBUSxDQUFDLENBQUMsR0FBYSxFQUFFLEVBQUU7WUFDNUIsSUFBSSxHQUFHLFlBQVksSUFBSSxFQUFFO2dCQUNyQixJQUFJLEdBQUcsQ0FBQyxRQUFRLEVBQUU7b0JBQ2QsR0FBRyxDQUFDLFFBQVEsQ0FBQyxPQUFPLEVBQUUsQ0FBQztpQkFDMUI7Z0JBQ0QsSUFBSSxDQUFDLGVBQWUsQ0FBQyxHQUFHLENBQUMsUUFBUSxDQUFDLENBQUM7YUFDdEM7WUFDRCxJQUFJLE9BQVEsR0FBVyxDQUFDLE9BQU8sS0FBSyxVQUFVLEVBQUU7Z0JBQzVDLGFBQWE7Z0JBQ2IsR0FBRyxDQUFDLE9BQU8sRUFBRSxDQUFDO2FBQ2pCO1FBQ0wsQ0FBQyxDQUFDLENBQUM7SUFDUCxDQUFDOztBQTlDTSx5QkFBYSxHQUFHLENBQUMsS0FBSyxFQUFFLFVBQVUsRUFBRSxTQUFTLEVBQUUsV0FBVyxFQUFFLGFBQWEsRUFBRSxRQUFRLEVBQUUsT0FBTyxFQUFFLGNBQWMsRUFBRSxjQUFjLENBQUMsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7TWF0ZXJpYWwsIE1lc2gsIE9iamVjdDNEfSBmcm9tIFwidGhyZWVcIjtcclxuXHJcbmV4cG9ydCBjbGFzcyBPYmplY3RVdGlscyB7XHJcbiAgICBzdGF0aWMgbWF0ZXJpYWxQcm9wcyA9IFsnbWFwJywgJ2xpZ2h0TWFwJywgJ2J1bXBNYXAnLCAnbm9ybWFsTWFwJywgJ3NwZWN1bGFyTWFwJywgJ2Vudk1hcCcsICdhb01hcCcsICdyb3VnaG5lc3NNYXAnLCAnbWV0YWxuZXNzTWFwJ107XHJcblxyXG4gICAgcHVibGljIHN0YXRpYyBEaXNwb3NlTWF0ZXJpYWwobWF0ZXJpYWw6IE1hdGVyaWFsIHwgTWF0ZXJpYWxbXSk6IHZvaWQge1xyXG4gICAgICAgIGlmICghbWF0ZXJpYWwpIHtcclxuICAgICAgICAgICAgcmV0dXJuO1xyXG4gICAgICAgIH1cclxuICAgICAgICBpZiAoQXJyYXkuaXNBcnJheShtYXRlcmlhbCkpIHtcclxuICAgICAgICAgICAgbWF0ZXJpYWwuZm9yRWFjaChtdHJsID0+IHtcclxuICAgICAgICAgICAgICAgIHRoaXMubWF0ZXJpYWxQcm9wc1xyXG4gICAgICAgICAgICAgICAgICAgIC5maWx0ZXIobWFwVHlwZSA9PiBtdHJsW21hcFR5cGVdKVxyXG4gICAgICAgICAgICAgICAgICAgIC5mb3JFYWNoKG1hcFR5cGUgPT4gbXRybFttYXBUeXBlXS5kaXNwb3NlKCkpO1xyXG4gICAgICAgICAgICAgICAgbXRybC5kaXNwb3NlKCk7XHJcbiAgICAgICAgICAgIH0pO1xyXG4gICAgICAgIH0gZWxzZSB7XHJcbiAgICAgICAgICAgIHRoaXMubWF0ZXJpYWxQcm9wc1xyXG4gICAgICAgICAgICAgICAgLmZpbHRlcihtYXBUeXBlID0+IG1hdGVyaWFsW21hcFR5cGVdICYmIG1hdGVyaWFsW21hcFR5cGVdLmRpc3Bvc2UpXHJcbiAgICAgICAgICAgICAgICAuZm9yRWFjaChtYXBUeXBlID0+IG1hdGVyaWFsW21hcFR5cGVdLmRpc3Bvc2UoKSk7XHJcbiAgICAgICAgICAgIGlmIChtYXRlcmlhbC5kaXNwb3NlKSB7XHJcbiAgICAgICAgICAgICAgICBtYXRlcmlhbC5kaXNwb3NlKCk7XHJcbiAgICAgICAgICAgIH1cclxuICAgICAgICB9XHJcbiAgICB9XHJcblxyXG4gICAgcHVibGljIHN0YXRpYyBEaXNwb3NlT2JqZWN0KG9iamVjdDogT2JqZWN0M0QgfCBNZXNoKTogdm9pZCB7XHJcbiAgICAgICAgaWYgKCFvYmplY3QpIHtcclxuICAgICAgICAgICAgcmV0dXJuO1xyXG4gICAgICAgIH1cclxuICAgICAgICBpZiAob2JqZWN0LmNoaWxkcmVuLmxlbmd0aCkge1xyXG4gICAgICAgICAgICBvYmplY3QuY2hpbGRyZW4uZm9yRWFjaChjaGlsZCA9PiB0aGlzLkRpc3Bvc2VPYmplY3QoY2hpbGQpKTtcclxuICAgICAgICB9XHJcbiAgICAgICAgdGhpcy5EaXNwb3NlTm9kZShvYmplY3QpO1xyXG4gICAgfVxyXG5cclxuICAgIHByaXZhdGUgc3RhdGljIERpc3Bvc2VOb2RlKG5vZGU6IE9iamVjdDNEIHwgTWVzaCk6IHZvaWQge1xyXG4gICAgICAgIG5vZGUudHJhdmVyc2UoKG9iajogT2JqZWN0M0QpID0+IHtcclxuICAgICAgICAgICAgaWYgKG9iaiBpbnN0YW5jZW9mIE1lc2gpIHtcclxuICAgICAgICAgICAgICAgIGlmIChvYmouZ2VvbWV0cnkpIHtcclxuICAgICAgICAgICAgICAgICAgICBvYmouZ2VvbWV0cnkuZGlzcG9zZSgpO1xyXG4gICAgICAgICAgICAgICAgfVxyXG4gICAgICAgICAgICAgICAgdGhpcy5EaXNwb3NlTWF0ZXJpYWwob2JqLm1hdGVyaWFsKTtcclxuICAgICAgICAgICAgfVxyXG4gICAgICAgICAgICBpZiAodHlwZW9mIChvYmogYXMgYW55KS5kaXNwb3NlID09PSAnZnVuY3Rpb24nKSB7XHJcbiAgICAgICAgICAgICAgICAvLyBAdHMtaWdub3JlXHJcbiAgICAgICAgICAgICAgICBvYmouZGlzcG9zZSgpO1xyXG4gICAgICAgICAgICB9XHJcbiAgICAgICAgfSk7XHJcbiAgICB9XHJcblxyXG59XHJcbiJdfQ==
|
|
@@ -1,94 +1,94 @@
|
|
|
1
|
-
import * as THREE from 'three';
|
|
2
|
-
export class SceneUtils {
|
|
3
|
-
static TrySelectorConnection(scene, parent, part1, part2, createAddableFn) {
|
|
4
|
-
let addable;
|
|
5
|
-
if (createAddableFn) {
|
|
6
|
-
addable = part2;
|
|
7
|
-
part2 = createAddableFn(addable);
|
|
8
|
-
}
|
|
9
|
-
for (let i = 0; i < part1.children.length; i++) {
|
|
10
|
-
for (let ii = 0; ii < part2.children.length; ii++) {
|
|
11
|
-
const con1 = part1.children[i];
|
|
12
|
-
const con2 = part2.children[ii];
|
|
13
|
-
// if these connectors can connect connect them
|
|
14
|
-
if (!con1['connected'] && !con2['connected'] && this.CanSelectorConnect(con1, con2)) {
|
|
15
|
-
// if addable, continue this loop with a new addable
|
|
16
|
-
if (addable) {
|
|
17
|
-
const newPart2 = createAddableFn(addable, part2, part1);
|
|
18
|
-
// reference of part2 changed, so refresh con2
|
|
19
|
-
this.SelectorConnect(scene, parent, con1, newPart2.children[ii]);
|
|
20
|
-
}
|
|
21
|
-
else {
|
|
22
|
-
return this.SelectorConnect(scene, parent, con1, con2);
|
|
23
|
-
}
|
|
24
|
-
}
|
|
25
|
-
}
|
|
26
|
-
}
|
|
27
|
-
return false;
|
|
28
|
-
}
|
|
29
|
-
static Convert3DPointToScreenPoint(point, camera, width, height) {
|
|
30
|
-
if (!point) {
|
|
31
|
-
return new THREE.Vector2();
|
|
32
|
-
}
|
|
33
|
-
const vector = point.clone().project(camera);
|
|
34
|
-
vector.x = (vector.x + 1) / 2 * width;
|
|
35
|
-
vector.y = -(vector.y - 1) / 2 * height;
|
|
36
|
-
return new THREE.Vector2(vector.x, vector.y);
|
|
37
|
-
}
|
|
38
|
-
static CanSelectorConnect(con1, con2) {
|
|
39
|
-
const con1Name = con1 instanceof THREE.Object3D ? con1.name : con1.connector;
|
|
40
|
-
const con2Name = con2 instanceof THREE.Object3D ? con2.name : con2.connector;
|
|
41
|
-
const sameParent = con1 instanceof THREE.Object3D && con2 instanceof THREE.Object3D ? con1.parent === con2.parent : false;
|
|
42
|
-
const parts1 = con1Name.toUpperCase().split('_'), parts2 = con2Name.toUpperCase().split('_');
|
|
43
|
-
if (parts1.length >= 3 && parts2.length >= 3 && !sameParent) {
|
|
44
|
-
const connectable = parts1[0] === 'C' && parts2[0] === 'C' &&
|
|
45
|
-
parts1[1] === parts2[1] && parts1[2] !== parts2[2];
|
|
46
|
-
return connectable;
|
|
47
|
-
}
|
|
48
|
-
return false;
|
|
49
|
-
}
|
|
50
|
-
static SelectorConnect(scene, parent, con1, con2) {
|
|
51
|
-
if (!scene || !parent) {
|
|
52
|
-
return false;
|
|
53
|
-
}
|
|
54
|
-
con1['connectedTo'] = con2.parent.name;
|
|
55
|
-
con2['connectedTo'] = con1.parent.name;
|
|
56
|
-
const motherRotation = new THREE.Euler(0, 0, 0);
|
|
57
|
-
const motherPosition = new THREE.Vector3(0, 0, 0);
|
|
58
|
-
scene.updateMatrixWorld(true);
|
|
59
|
-
motherRotation.copy(parent.rotation);
|
|
60
|
-
parent.getWorldPosition(motherPosition);
|
|
61
|
-
parent.rotation.set(0, 0, 0);
|
|
62
|
-
parent.position.set(0, 0, 0);
|
|
63
|
-
const con1Quat = new THREE.Quaternion();
|
|
64
|
-
con1.getWorldQuaternion(con1Quat);
|
|
65
|
-
const con2Quat = new THREE.Quaternion();
|
|
66
|
-
con2.getWorldQuaternion(con2Quat);
|
|
67
|
-
const con1QuatParent = new THREE.Quaternion();
|
|
68
|
-
con1.parent.getWorldQuaternion(con1QuatParent);
|
|
69
|
-
const con2QuatParent = new THREE.Quaternion();
|
|
70
|
-
con2.parent.getWorldQuaternion(con2QuatParent);
|
|
71
|
-
const rotation = new THREE.Quaternion().multiplyQuaternions(con1Quat.invert(), con2Quat).multiply(new THREE.Quaternion().multiplyQuaternions(con1QuatParent, con1QuatParent));
|
|
72
|
-
con2.parent.quaternion.copy(rotation);
|
|
73
|
-
// Update because the matrix has been tempered with
|
|
74
|
-
scene.updateMatrixWorld(true);
|
|
75
|
-
// Move the connectors towards eachother
|
|
76
|
-
const con1Pos = new THREE.Vector3();
|
|
77
|
-
con1.getWorldPosition(con1Pos);
|
|
78
|
-
const con2Pos = new THREE.Vector3();
|
|
79
|
-
con2.getWorldPosition(con2Pos);
|
|
80
|
-
const move = con1Pos.sub(con2Pos);
|
|
81
|
-
con2.parent.position.x += move.x;
|
|
82
|
-
con2.parent.position.y += move.y;
|
|
83
|
-
con2.parent.position.z += move.z;
|
|
84
|
-
// reset parent's objects rotation and position
|
|
85
|
-
parent.rotation.copy(motherRotation);
|
|
86
|
-
parent.position.copy(motherPosition);
|
|
87
|
-
scene.updateMatrixWorld(true);
|
|
88
|
-
// Set the connected flag
|
|
89
|
-
con1['connected'] = true;
|
|
90
|
-
con2['connected'] = true;
|
|
91
|
-
return true;
|
|
92
|
-
}
|
|
93
|
-
}
|
|
1
|
+
import * as THREE from 'three';
|
|
2
|
+
export class SceneUtils {
|
|
3
|
+
static TrySelectorConnection(scene, parent, part1, part2, createAddableFn) {
|
|
4
|
+
let addable;
|
|
5
|
+
if (createAddableFn) {
|
|
6
|
+
addable = part2;
|
|
7
|
+
part2 = createAddableFn(addable);
|
|
8
|
+
}
|
|
9
|
+
for (let i = 0; i < part1.children.length; i++) {
|
|
10
|
+
for (let ii = 0; ii < part2.children.length; ii++) {
|
|
11
|
+
const con1 = part1.children[i];
|
|
12
|
+
const con2 = part2.children[ii];
|
|
13
|
+
// if these connectors can connect connect them
|
|
14
|
+
if (!con1['connected'] && !con2['connected'] && this.CanSelectorConnect(con1, con2)) {
|
|
15
|
+
// if addable, continue this loop with a new addable
|
|
16
|
+
if (addable) {
|
|
17
|
+
const newPart2 = createAddableFn(addable, part2, part1);
|
|
18
|
+
// reference of part2 changed, so refresh con2
|
|
19
|
+
this.SelectorConnect(scene, parent, con1, newPart2.children[ii]);
|
|
20
|
+
}
|
|
21
|
+
else {
|
|
22
|
+
return this.SelectorConnect(scene, parent, con1, con2);
|
|
23
|
+
}
|
|
24
|
+
}
|
|
25
|
+
}
|
|
26
|
+
}
|
|
27
|
+
return false;
|
|
28
|
+
}
|
|
29
|
+
static Convert3DPointToScreenPoint(point, camera, width, height) {
|
|
30
|
+
if (!point) {
|
|
31
|
+
return new THREE.Vector2();
|
|
32
|
+
}
|
|
33
|
+
const vector = point.clone().project(camera);
|
|
34
|
+
vector.x = (vector.x + 1) / 2 * width;
|
|
35
|
+
vector.y = -(vector.y - 1) / 2 * height;
|
|
36
|
+
return new THREE.Vector2(vector.x, vector.y);
|
|
37
|
+
}
|
|
38
|
+
static CanSelectorConnect(con1, con2) {
|
|
39
|
+
const con1Name = con1 instanceof THREE.Object3D ? con1.name : con1.connector;
|
|
40
|
+
const con2Name = con2 instanceof THREE.Object3D ? con2.name : con2.connector;
|
|
41
|
+
const sameParent = con1 instanceof THREE.Object3D && con2 instanceof THREE.Object3D ? con1.parent === con2.parent : false;
|
|
42
|
+
const parts1 = con1Name.toUpperCase().split('_'), parts2 = con2Name.toUpperCase().split('_');
|
|
43
|
+
if (parts1.length >= 3 && parts2.length >= 3 && !sameParent) {
|
|
44
|
+
const connectable = parts1[0] === 'C' && parts2[0] === 'C' &&
|
|
45
|
+
parts1[1] === parts2[1] && parts1[2] !== parts2[2];
|
|
46
|
+
return connectable;
|
|
47
|
+
}
|
|
48
|
+
return false;
|
|
49
|
+
}
|
|
50
|
+
static SelectorConnect(scene, parent, con1, con2) {
|
|
51
|
+
if (!scene || !parent) {
|
|
52
|
+
return false;
|
|
53
|
+
}
|
|
54
|
+
con1['connectedTo'] = con2.parent.name;
|
|
55
|
+
con2['connectedTo'] = con1.parent.name;
|
|
56
|
+
const motherRotation = new THREE.Euler(0, 0, 0);
|
|
57
|
+
const motherPosition = new THREE.Vector3(0, 0, 0);
|
|
58
|
+
scene.updateMatrixWorld(true);
|
|
59
|
+
motherRotation.copy(parent.rotation);
|
|
60
|
+
parent.getWorldPosition(motherPosition);
|
|
61
|
+
parent.rotation.set(0, 0, 0);
|
|
62
|
+
parent.position.set(0, 0, 0);
|
|
63
|
+
const con1Quat = new THREE.Quaternion();
|
|
64
|
+
con1.getWorldQuaternion(con1Quat);
|
|
65
|
+
const con2Quat = new THREE.Quaternion();
|
|
66
|
+
con2.getWorldQuaternion(con2Quat);
|
|
67
|
+
const con1QuatParent = new THREE.Quaternion();
|
|
68
|
+
con1.parent.getWorldQuaternion(con1QuatParent);
|
|
69
|
+
const con2QuatParent = new THREE.Quaternion();
|
|
70
|
+
con2.parent.getWorldQuaternion(con2QuatParent);
|
|
71
|
+
const rotation = new THREE.Quaternion().multiplyQuaternions(con1Quat.invert(), con2Quat).multiply(new THREE.Quaternion().multiplyQuaternions(con1QuatParent, con1QuatParent));
|
|
72
|
+
con2.parent.quaternion.copy(rotation);
|
|
73
|
+
// Update because the matrix has been tempered with
|
|
74
|
+
scene.updateMatrixWorld(true);
|
|
75
|
+
// Move the connectors towards eachother
|
|
76
|
+
const con1Pos = new THREE.Vector3();
|
|
77
|
+
con1.getWorldPosition(con1Pos);
|
|
78
|
+
const con2Pos = new THREE.Vector3();
|
|
79
|
+
con2.getWorldPosition(con2Pos);
|
|
80
|
+
const move = con1Pos.sub(con2Pos);
|
|
81
|
+
con2.parent.position.x += move.x;
|
|
82
|
+
con2.parent.position.y += move.y;
|
|
83
|
+
con2.parent.position.z += move.z;
|
|
84
|
+
// reset parent's objects rotation and position
|
|
85
|
+
parent.rotation.copy(motherRotation);
|
|
86
|
+
parent.position.copy(motherPosition);
|
|
87
|
+
scene.updateMatrixWorld(true);
|
|
88
|
+
// Set the connected flag
|
|
89
|
+
con1['connected'] = true;
|
|
90
|
+
con2['connected'] = true;
|
|
91
|
+
return true;
|
|
92
|
+
}
|
|
93
|
+
}
|
|
94
94
|
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"scene-utils.js","sourceRoot":"ng://@colijnit/configurator/","sources":["utils/scene-utils.ts"],"names":[],"mappings":"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","sourcesContent":["import * as THREE from 'three';\r\n\r\nexport class SceneUtils {\r\n\r\n    public static TrySelectorConnection(scene: THREE.Scene, parent: THREE.Object3D,\r\n        part1: any, part2: any, createAddableFn?: Function): boolean {\r\n        let addable: any;\r\n        if (createAddableFn) {\r\n            addable = part2;\r\n            part2 = createAddableFn(addable);\r\n        }\r\n        for (let i = 0; i < part1.children.length; i++) {\r\n            for (let ii = 0; ii < part2.children.length; ii++) {\r\n                const con1: THREE.Object3D = part1.children[i];\r\n                const con2: THREE.Object3D = part2.children[ii];\r\n                // if these connectors can connect connect them\r\n                if (!con1['connected'] && !con2['connected'] && this.CanSelectorConnect(con1, con2)) {\r\n                    // if addable, continue this loop with a new addable\r\n                    if (addable) {\r\n                        const newPart2: THREE.Object3D = createAddableFn(addable, part2, part1);\r\n                        // reference of part2 changed, so refresh con2\r\n                        this.SelectorConnect(scene, parent, con1, newPart2.children[ii]);\r\n                    } else {\r\n                        return this.SelectorConnect(scene, parent, con1, con2);\r\n                    }\r\n                }\r\n            }\r\n        }\r\n        return false;\r\n    }\r\n\r\n    public static Convert3DPointToScreenPoint(point: THREE.Vector3, camera: THREE.Camera, width: number, height: number): THREE.Vector2 {\r\n        if (!point) {\r\n            return new THREE.Vector2();\r\n        }\r\n        const vector: THREE.Vector3 = point.clone().project(camera);\r\n        vector.x = (vector.x + 1) / 2 * width;\r\n        vector.y = -(vector.y - 1) / 2 * height;\r\n        return new THREE.Vector2(vector.x, vector.y);\r\n    }\r\n\r\n    public static CanSelectorConnect(con1: THREE.Object3D | any, con2: THREE.Object3D | any): boolean {\r\n        const con1Name: string = con1 instanceof THREE.Object3D ? con1.name : con1.connector;\r\n        const con2Name: string = con2 instanceof THREE.Object3D ? con2.name : con2.connector;\r\n        const sameParent: boolean = con1 instanceof THREE.Object3D && con2 instanceof THREE.Object3D ? con1.parent === con2.parent : false;\r\n        const parts1: string[] = con1Name.toUpperCase().split('_'),\r\n            parts2: string[] = con2Name.toUpperCase().split('_');\r\n        if (parts1.length >= 3 && parts2.length >= 3 && !sameParent) {\r\n            const connectable: boolean = parts1[0] === 'C' && parts2[0] === 'C' &&\r\n                parts1[1] === parts2[1] && parts1[2] !== parts2[2];\r\n            return connectable;\r\n        }\r\n        return false;\r\n    }\r\n\r\n    private static SelectorConnect(scene: THREE.Scene, parent: THREE.Object3D, con1: THREE.Object3D, con2: THREE.Object3D): boolean {\r\n        if (!scene || !parent) {\r\n            return false;\r\n        }\r\n        con1['connectedTo'] = con2.parent.name;\r\n        con2['connectedTo'] = con1.parent.name;\r\n\r\n        const motherRotation: THREE.Euler = new THREE.Euler(0, 0, 0);\r\n        const motherPosition: THREE.Vector3 = new THREE.Vector3(0, 0, 0);\r\n\r\n        scene.updateMatrixWorld(true);\r\n        motherRotation.copy(parent.rotation);\r\n        parent.getWorldPosition(motherPosition);\r\n\r\n        parent.rotation.set(0, 0, 0);\r\n        parent.position.set(0, 0, 0);\r\n\r\n        const con1Quat: THREE.Quaternion = new THREE.Quaternion();\r\n        con1.getWorldQuaternion(con1Quat);\r\n        const con2Quat: THREE.Quaternion = new THREE.Quaternion();\r\n        con2.getWorldQuaternion(con2Quat);\r\n        const con1QuatParent: THREE.Quaternion = new THREE.Quaternion();\r\n        con1.parent.getWorldQuaternion(con1QuatParent);\r\n        const con2QuatParent: THREE.Quaternion = new THREE.Quaternion();\r\n        con2.parent.getWorldQuaternion(con2QuatParent);\r\n\r\n        const rotation: THREE.Quaternion = new THREE.Quaternion().multiplyQuaternions(\r\n            con1Quat.invert(),\r\n            con2Quat\r\n        ).multiply(\r\n            new THREE.Quaternion().multiplyQuaternions(\r\n                con1QuatParent,\r\n                con1QuatParent\r\n            )\r\n        );\r\n\r\n        con2.parent.quaternion.copy(rotation);\r\n        // Update because the matrix has been tempered with\r\n        scene.updateMatrixWorld(true);\r\n\r\n        // Move the connectors towards eachother\r\n        const con1Pos: THREE.Vector3 = new THREE.Vector3();\r\n        con1.getWorldPosition(con1Pos);\r\n        const con2Pos: THREE.Vector3 = new THREE.Vector3();\r\n        con2.getWorldPosition(con2Pos);\r\n\r\n        const move: THREE.Vector3 = con1Pos.sub(con2Pos);\r\n        con2.parent.position.x += move.x;\r\n        con2.parent.position.y += move.y;\r\n        con2.parent.position.z += move.z;\r\n\r\n        // reset parent's objects rotation and position\r\n        parent.rotation.copy(motherRotation);\r\n        parent.position.copy(motherPosition);\r\n\r\n        scene.updateMatrixWorld(true);\r\n\r\n        // Set the connected flag\r\n        con1['connected'] = true;\r\n        con2['connected'] = true;\r\n        return true;\r\n    }\r\n\r\n}\r\n"]}
|