fx-form-builder-wrapper 2.0.58 → 2.0.61
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/esm2022/lib/components/three-viewer/three-viewer.component.mjs +161 -0
- package/esm2022/lib/components/uploader/uploader.component.mjs +101 -11
- package/fesm2022/fx-form-builder-wrapper.mjs +273 -27
- package/fesm2022/fx-form-builder-wrapper.mjs.map +1 -1
- package/lib/components/three-viewer/three-viewer.component.d.ts +30 -0
- package/lib/components/uploader/uploader.component.d.ts +17 -2
- package/package.json +1 -1
|
@@ -0,0 +1,161 @@
|
|
|
1
|
+
import { Component, ViewChild, Input } from '@angular/core';
|
|
2
|
+
import * as THREE from 'three';
|
|
3
|
+
import { STLLoader } from 'three/examples/jsm/loaders/STLLoader.js';
|
|
4
|
+
import * as i0 from "@angular/core";
|
|
5
|
+
import * as i1 from "@angular/common/http";
|
|
6
|
+
export class ThreeViewerComponent {
|
|
7
|
+
http;
|
|
8
|
+
viewerContainer;
|
|
9
|
+
file;
|
|
10
|
+
manualFileUpload = null;
|
|
11
|
+
scene;
|
|
12
|
+
camera;
|
|
13
|
+
renderer;
|
|
14
|
+
loader = new STLLoader();
|
|
15
|
+
model;
|
|
16
|
+
isDragging = false;
|
|
17
|
+
previousMousePosition = { x: 0, y: 0 };
|
|
18
|
+
zoomSpeed = 1.1;
|
|
19
|
+
constructor(http) {
|
|
20
|
+
this.http = http;
|
|
21
|
+
}
|
|
22
|
+
ngAfterViewInit() {
|
|
23
|
+
this.initThree();
|
|
24
|
+
}
|
|
25
|
+
ngOnChanges(changes) {
|
|
26
|
+
if (changes['manualFileUpload']?.currentValue && this.manualFileUpload) {
|
|
27
|
+
this.loadModelFromBlob(this.manualFileUpload);
|
|
28
|
+
}
|
|
29
|
+
}
|
|
30
|
+
ngOnDestroy() {
|
|
31
|
+
this.cleanupThreeJS();
|
|
32
|
+
}
|
|
33
|
+
initThree() {
|
|
34
|
+
const container = this.viewerContainer.nativeElement;
|
|
35
|
+
this.scene = new THREE.Scene();
|
|
36
|
+
this.scene.background = new THREE.Color(0xf0f0f0);
|
|
37
|
+
this.camera = new THREE.PerspectiveCamera(60, container.clientWidth / container.clientHeight, 0.1, 1000);
|
|
38
|
+
this.camera.position.set(0, 0, 100);
|
|
39
|
+
this.renderer = new THREE.WebGLRenderer({ antialias: true });
|
|
40
|
+
this.renderer.setSize(container.clientWidth, container.clientHeight);
|
|
41
|
+
container.appendChild(this.renderer.domElement);
|
|
42
|
+
const ambientLight = new THREE.AmbientLight(0x404040, 2);
|
|
43
|
+
this.scene.add(ambientLight);
|
|
44
|
+
const directionalLight = new THREE.DirectionalLight(0xffffff, 1);
|
|
45
|
+
directionalLight.position.set(1, 1, 1).normalize();
|
|
46
|
+
this.scene.add(directionalLight);
|
|
47
|
+
this.initControls();
|
|
48
|
+
this.animate();
|
|
49
|
+
}
|
|
50
|
+
initControls() {
|
|
51
|
+
const canvas = this.renderer.domElement;
|
|
52
|
+
// Mouse Drag for Rotation
|
|
53
|
+
canvas.addEventListener('mousedown', (event) => {
|
|
54
|
+
this.isDragging = true;
|
|
55
|
+
this.previousMousePosition.x = event.clientX;
|
|
56
|
+
this.previousMousePosition.y = event.clientY;
|
|
57
|
+
});
|
|
58
|
+
canvas.addEventListener('mouseup', () => {
|
|
59
|
+
this.isDragging = false;
|
|
60
|
+
});
|
|
61
|
+
canvas.addEventListener('mousemove', (event) => {
|
|
62
|
+
if (!this.isDragging)
|
|
63
|
+
return;
|
|
64
|
+
const deltaX = event.clientX - this.previousMousePosition.x;
|
|
65
|
+
const deltaY = event.clientY - this.previousMousePosition.y;
|
|
66
|
+
const rotationSpeed = 0.005;
|
|
67
|
+
this.model?.rotateY(deltaX * rotationSpeed);
|
|
68
|
+
this.model?.rotateX(deltaY * rotationSpeed);
|
|
69
|
+
this.previousMousePosition.x = event.clientX;
|
|
70
|
+
this.previousMousePosition.y = event.clientY;
|
|
71
|
+
});
|
|
72
|
+
// Scroll for Zoom
|
|
73
|
+
canvas.addEventListener('wheel', (event) => {
|
|
74
|
+
// event.preventDefault();
|
|
75
|
+
if (event.deltaY < 0) {
|
|
76
|
+
this.camera.position.multiplyScalar(1 / this.zoomSpeed); // Zoom In
|
|
77
|
+
}
|
|
78
|
+
else {
|
|
79
|
+
this.camera.position.multiplyScalar(this.zoomSpeed); // Zoom Out
|
|
80
|
+
}
|
|
81
|
+
this.camera.updateProjectionMatrix();
|
|
82
|
+
});
|
|
83
|
+
// Resize Handling
|
|
84
|
+
window.addEventListener('resize', () => this.onWindowResize());
|
|
85
|
+
}
|
|
86
|
+
loadModelFromBlob(blob) {
|
|
87
|
+
const reader = new FileReader();
|
|
88
|
+
reader.onload = () => {
|
|
89
|
+
const arrayBuffer = reader.result;
|
|
90
|
+
const geometry = this.loader.parse(arrayBuffer);
|
|
91
|
+
geometry.computeVertexNormals();
|
|
92
|
+
if (this.model) {
|
|
93
|
+
this.scene.remove(this.model);
|
|
94
|
+
this.model.geometry.dispose();
|
|
95
|
+
}
|
|
96
|
+
const material = new THREE.MeshPhysicalMaterial({
|
|
97
|
+
color: 0xF8F0E3,
|
|
98
|
+
metalness: 0.1,
|
|
99
|
+
roughness: 0.5,
|
|
100
|
+
transmission: 0.5, // Creates slight translucency
|
|
101
|
+
clearcoat: 1.0, // Adds enamel-like reflection
|
|
102
|
+
clearcoatRoughness: 0.3
|
|
103
|
+
});
|
|
104
|
+
this.model = new THREE.Mesh(geometry, material);
|
|
105
|
+
this.scene.add(this.model);
|
|
106
|
+
this.centerModel();
|
|
107
|
+
};
|
|
108
|
+
reader.readAsArrayBuffer(blob);
|
|
109
|
+
}
|
|
110
|
+
centerModel() {
|
|
111
|
+
if (!this.model)
|
|
112
|
+
return;
|
|
113
|
+
const box = new THREE.Box3().setFromObject(this.model);
|
|
114
|
+
const center = new THREE.Vector3();
|
|
115
|
+
box.getCenter(center);
|
|
116
|
+
this.model.position.sub(center);
|
|
117
|
+
const size = box.getSize(new THREE.Vector3()).length();
|
|
118
|
+
this.camera.position.set(0, 0, size * 2);
|
|
119
|
+
this.camera.updateProjectionMatrix();
|
|
120
|
+
this.onWindowResize();
|
|
121
|
+
}
|
|
122
|
+
animate = () => {
|
|
123
|
+
requestAnimationFrame(this.animate);
|
|
124
|
+
this.renderer.render(this.scene, this.camera);
|
|
125
|
+
};
|
|
126
|
+
onWindowResize() {
|
|
127
|
+
const container = this.viewerContainer.nativeElement;
|
|
128
|
+
this.camera.aspect = container.clientWidth / container.clientHeight;
|
|
129
|
+
this.camera.updateProjectionMatrix();
|
|
130
|
+
this.renderer.setSize(container.clientWidth, container.clientHeight);
|
|
131
|
+
}
|
|
132
|
+
cleanupThreeJS() {
|
|
133
|
+
// Remove event listeners
|
|
134
|
+
const canvas = this.renderer.domElement;
|
|
135
|
+
canvas.removeEventListener('mousedown', () => { });
|
|
136
|
+
canvas.removeEventListener('mouseup', () => { });
|
|
137
|
+
canvas.removeEventListener('mousemove', () => { });
|
|
138
|
+
canvas.removeEventListener('wheel', () => { });
|
|
139
|
+
window.removeEventListener('resize', () => this.onWindowResize());
|
|
140
|
+
// Dispose of Three.js resources
|
|
141
|
+
if (this.model) {
|
|
142
|
+
this.scene.remove(this.model);
|
|
143
|
+
this.model.geometry.dispose();
|
|
144
|
+
}
|
|
145
|
+
this.renderer.dispose();
|
|
146
|
+
}
|
|
147
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ThreeViewerComponent, deps: [{ token: i1.HttpClient }], target: i0.ɵɵFactoryTarget.Component });
|
|
148
|
+
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: ThreeViewerComponent, isStandalone: true, selector: "app-three-viewer", inputs: { file: "file", manualFileUpload: "manualFileUpload" }, viewQueries: [{ propertyName: "viewerContainer", first: true, predicate: ["viewerContainer"], descendants: true, static: true }], usesOnChanges: true, ngImport: i0, template: "<div #viewerContainer class=\"viewer-container\"></div>\r\n ", styles: [".viewer-container{position:relative;width:100%;max-height:500px;overflow:hidden}button{position:absolute;top:10px;background:#000000b3;color:#fff;border:none;padding:5px 10px;cursor:pointer;margin:5px}button:hover{background:#000000e6}::ng-deep canvas{display:block;width:100%!important;height:100%!important;min-height:450px!important;min-width:986px!important}@media (min-width: 1600px){.viewer-container{width:100%;max-height:650px}canvas{min-width:1326px!important}}\n"] });
|
|
149
|
+
}
|
|
150
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ThreeViewerComponent, decorators: [{
|
|
151
|
+
type: Component,
|
|
152
|
+
args: [{ selector: 'app-three-viewer', standalone: true, template: "<div #viewerContainer class=\"viewer-container\"></div>\r\n ", styles: [".viewer-container{position:relative;width:100%;max-height:500px;overflow:hidden}button{position:absolute;top:10px;background:#000000b3;color:#fff;border:none;padding:5px 10px;cursor:pointer;margin:5px}button:hover{background:#000000e6}::ng-deep canvas{display:block;width:100%!important;height:100%!important;min-height:450px!important;min-width:986px!important}@media (min-width: 1600px){.viewer-container{width:100%;max-height:650px}canvas{min-width:1326px!important}}\n"] }]
|
|
153
|
+
}], ctorParameters: () => [{ type: i1.HttpClient }], propDecorators: { viewerContainer: [{
|
|
154
|
+
type: ViewChild,
|
|
155
|
+
args: ['viewerContainer', { static: true }]
|
|
156
|
+
}], file: [{
|
|
157
|
+
type: Input
|
|
158
|
+
}], manualFileUpload: [{
|
|
159
|
+
type: Input
|
|
160
|
+
}] } });
|
|
161
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"three-viewer.component.js","sourceRoot":"","sources":["../../../../../../projects/fx-builder-wrapper/src/lib/components/three-viewer/three-viewer.component.ts","../../../../../../projects/fx-builder-wrapper/src/lib/components/three-viewer/three-viewer.component.html"],"names":[],"mappings":"AAAA,OAAO,EACL,SAAS,EAA6B,SAAS,EAAE,KAAK,EACvD,MAAM,eAAe,CAAC;AAEvB,OAAO,KAAK,KAAK,MAAM,OAAO,CAAC;AAC/B,OAAO,EAAE,SAAS,EAAE,MAAM,yCAAyC,CAAC;;;AAQpE,MAAM,OAAO,oBAAoB;IAeX;IAd4B,eAAe,CAAc;IACpE,IAAI,CAAO;IACX,gBAAgB,GAAgB,IAAI,CAAC;IAEtC,KAAK,CAAe;IACpB,MAAM,CAA2B;IACjC,QAAQ,CAAuB;IAC/B,MAAM,GAAG,IAAI,SAAS,EAAE,CAAC;IACzB,KAAK,CAAqB;IAE1B,UAAU,GAAG,KAAK,CAAC;IACnB,qBAAqB,GAAG,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC;IACvC,SAAS,GAAG,GAAG,CAAC;IAExB,YAAoB,IAAgB;QAAhB,SAAI,GAAJ,IAAI,CAAY;IAAG,CAAC;IAExC,eAAe;QACb,IAAI,CAAC,SAAS,EAAE,CAAC;IACnB,CAAC;IAED,WAAW,CAAC,OAAsB;QAChC,IAAI,OAAO,CAAC,kBAAkB,CAAC,EAAE,YAAY,IAAI,IAAI,CAAC,gBAAgB,EAAE,CAAC;YACvE,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,gBAAgB,CAAC,CAAC;QAChD,CAAC;IACH,CAAC;IAED,WAAW;QACT,IAAI,CAAC,cAAc,EAAE,CAAC;IACxB,CAAC;IAEO,SAAS;QACf,MAAM,SAAS,GAAG,IAAI,CAAC,eAAe,CAAC,aAAa,CAAC;QACrD,IAAI,CAAC,KAAK,GAAG,IAAI,KAAK,CAAC,KAAK,EAAE,CAAC;QAC/B,IAAI,CAAC,KAAK,CAAC,UAAU,GAAG,IAAI,KAAK,CAAC,KAAK,CAAC,QAAQ,CAAC,CAAC;QAElD,IAAI,CAAC,MAAM,GAAG,IAAI,KAAK,CAAC,iBAAiB,CAAC,EAAE,EAAE,SAAS,CAAC,WAAW,GAAG,SAAS,CAAC,YAAY,EAAE,GAAG,EAAE,IAAI,CAAC,CAAC;QACzG,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC,EAAE,CAAC,EAAE,GAAG,CAAC,CAAC;QAEpC,IAAI,CAAC,QAAQ,GAAG,IAAI,KAAK,CAAC,aAAa,CAAC,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC,CAAC;QAC7D,IAAI,CAAC,QAAQ,CAAC,OAAO,CAAC,SAAS,CAAC,WAAW,EAAE,SAAS,CAAC,YAAY,CAAC,CAAC;QACrE,SAAS,CAAC,WAAW,CAAC,IAAI,CAAC,QAAQ,CAAC,UAAU,CAAC,CAAC;QAEhD,MAAM,YAAY,GAAG,IAAI,KAAK,CAAC,YAAY,CAAC,QAAQ,EAAE,CAAC,CAAC,CAAC;QACzD,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,YAAY,CAAC,CAAC;QAE7B,MAAM,gBAAgB,GAAG,IAAI,KAAK,CAAC,gBAAgB,CAAC,QAAQ,EAAE,CAAC,CAAC,CAAC;QACjE,gBAAgB,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC,EAAE,CAAC,EAAE,CAAC,CAAC,CAAC,SAAS,EAAE,CAAC;QACnD,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,gBAAgB,CAAC,CAAC;QAEjC,IAAI,CAAC,YAAY,EAAE,CAAC;QACpB,IAAI,CAAC,OAAO,EAAE,CAAC;IACjB,CAAC;IAEO,YAAY;QAClB,MAAM,MAAM,GAAG,IAAI,CAAC,QAAQ,CAAC,UAAU,CAAC;QAExC,0BAA0B;QAC1B,MAAM,CAAC,gBAAgB,CAAC,WAAW,EAAE,CAAC,KAAK,EAAE,EAAE;YAC7C,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC;YACvB,IAAI,CAAC,qBAAqB,CAAC,CAAC,GAAG,KAAK,CAAC,OAAO,CAAC;YAC7C,IAAI,CAAC,qBAAqB,CAAC,CAAC,GAAG,KAAK,CAAC,OAAO,CAAC;QAC/C,CAAC,CAAC,CAAC;QAEH,MAAM,CAAC,gBAAgB,CAAC,SAAS,EAAE,GAAG,EAAE;YACtC,IAAI,CAAC,UAAU,GAAG,KAAK,CAAC;QAC1B,CAAC,CAAC,CAAC;QAEH,MAAM,CAAC,gBAAgB,CAAC,WAAW,EAAE,CAAC,KAAK,EAAE,EAAE;YAC7C,IAAI,CAAC,IAAI,CAAC,UAAU;gBAAE,OAAO;YAE7B,MAAM,MAAM,GAAG,KAAK,CAAC,OAAO,GAAG,IAAI,CAAC,qBAAqB,CAAC,CAAC,CAAC;YAC5D,MAAM,MAAM,GAAG,KAAK,CAAC,OAAO,GAAG,IAAI,CAAC,qBAAqB,CAAC,CAAC,CAAC;YAE5D,MAAM,aAAa,GAAG,KAAK,CAAC;YAC5B,IAAI,CAAC,KAAK,EAAE,OAAO,CAAC,MAAM,GAAG,aAAa,CAAC,CAAC;YAC5C,IAAI,CAAC,KAAK,EAAE,OAAO,CAAC,MAAM,GAAG,aAAa,CAAC,CAAC;YAE5C,IAAI,CAAC,qBAAqB,CAAC,CAAC,GAAG,KAAK,CAAC,OAAO,CAAC;YAC7C,IAAI,CAAC,qBAAqB,CAAC,CAAC,GAAG,KAAK,CAAC,OAAO,CAAC;QAC/C,CAAC,CAAC,CAAC;QAEH,kBAAkB;QAClB,MAAM,CAAC,gBAAgB,CAAC,OAAO,EAAE,CAAC,KAAK,EAAE,EAAE;YACzC,0BAA0B;YAC1B,IAAI,KAAK,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;gBACrB,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,cAAc,CAAC,CAAC,GAAG,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC,UAAU;YACrE,CAAC;iBAAM,CAAC;gBACN,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,cAAc,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC,WAAW;YAClE,CAAC;YACD,IAAI,CAAC,MAAM,CAAC,sBAAsB,EAAE,CAAC;QACvC,CAAC,CAAC,CAAC;QAEH,kBAAkB;QAClB,MAAM,CAAC,gBAAgB,CAAC,QAAQ,EAAE,GAAG,EAAE,CAAC,IAAI,CAAC,cAAc,EAAE,CAAC,CAAC;IACjE,CAAC;IAEO,iBAAiB,CAAC,IAAU;QAClC,MAAM,MAAM,GAAG,IAAI,UAAU,EAAE,CAAC;QAChC,MAAM,CAAC,MAAM,GAAG,GAAG,EAAE;YACnB,MAAM,WAAW,GAAG,MAAM,CAAC,MAAqB,CAAC;YACjD,MAAM,QAAQ,GAAG,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,WAAW,CAAC,CAAC;YAChD,QAAQ,CAAC,oBAAoB,EAAE,CAAC;YAEhC,IAAI,IAAI,CAAC,KAAK,EAAE,CAAC;gBACf,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;gBAC9B,IAAI,CAAC,KAAK,CAAC,QAAQ,CAAC,OAAO,EAAE,CAAC;YAChC,CAAC;YAED,MAAM,QAAQ,GAAG,IAAI,KAAK,CAAC,oBAAoB,CAAC;gBAC9C,KAAK,EAAE,QAAQ;gBACf,SAAS,EAAE,GAAG;gBACd,SAAS,EAAE,GAAG;gBACd,YAAY,EAAE,GAAG,EAAE,8BAA8B;gBACjD,SAAS,EAAE,GAAG,EAAK,8BAA8B;gBACjD,kBAAkB,EAAE,GAAG;aACxB,CAAC,CAAC;YAEH,IAAI,CAAC,KAAK,GAAG,IAAI,KAAK,CAAC,IAAI,CAAC,QAAQ,EAAE,QAAQ,CAAC,CAAC;YAChD,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;YAE3B,IAAI,CAAC,WAAW,EAAE,CAAC;QACrB,CAAC,CAAC;QACF,MAAM,CAAC,iBAAiB,CAAC,IAAI,CAAC,CAAC;IACjC,CAAC;IAEO,WAAW;QACjB,IAAI,CAAC,IAAI,CAAC,KAAK;YAAE,OAAO;QACxB,MAAM,GAAG,GAAG,IAAI,KAAK,CAAC,IAAI,EAAE,CAAC,aAAa,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QACvD,MAAM,MAAM,GAAG,IAAI,KAAK,CAAC,OAAO,EAAE,CAAC;QACnC,GAAG,CAAC,SAAS,CAAC,MAAM,CAAC,CAAC;QACtB,IAAI,CAAC,KAAK,CAAC,QAAQ,CAAC,GAAG,CAAC,MAAM,CAAC,CAAC;QAEhC,MAAM,IAAI,GAAG,GAAG,CAAC,OAAO,CAAC,IAAI,KAAK,CAAC,OAAO,EAAE,CAAC,CAAC,MAAM,EAAE,CAAC;QACvD,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC,EAAE,CAAC,EAAE,IAAI,GAAG,CAAC,CAAC,CAAC;QACzC,IAAI,CAAC,MAAM,CAAC,sBAAsB,EAAE,CAAC;QACrC,IAAI,CAAC,cAAc,EAAE,CAAC;IACxB,CAAC;IAEO,OAAO,GAAG,GAAS,EAAE;QAC3B,qBAAqB,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;QACpC,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC,IAAI,CAAC,KAAK,EAAE,IAAI,CAAC,MAAM,CAAC,CAAC;IAChD,CAAC,CAAC;IAEF,cAAc;QACZ,MAAM,SAAS,GAAG,IAAI,CAAC,eAAe,CAAC,aAAa,CAAC;QACrD,IAAI,CAAC,MAAM,CAAC,MAAM,GAAG,SAAS,CAAC,WAAW,GAAG,SAAS,CAAC,YAAY,CAAC;QACpE,IAAI,CAAC,MAAM,CAAC,sBAAsB,EAAE,CAAC;QACrC,IAAI,CAAC,QAAQ,CAAC,OAAO,CAAC,SAAS,CAAC,WAAW,EAAE,SAAS,CAAC,YAAY,CAAC,CAAC;IACvE,CAAC;IAEO,cAAc;QACpB,yBAAyB;QACzB,MAAM,MAAM,GAAG,IAAI,CAAC,QAAQ,CAAC,UAAU,CAAC;QACxC,MAAM,CAAC,mBAAmB,CAAC,WAAW,EAAE,GAAG,EAAE,GAAE,CAAC,CAAC,CAAC;QAClD,MAAM,CAAC,mBAAmB,CAAC,SAAS,EAAE,GAAG,EAAE,GAAE,CAAC,CAAC,CAAC;QAChD,MAAM,CAAC,mBAAmB,CAAC,WAAW,EAAE,GAAG,EAAE,GAAE,CAAC,CAAC,CAAC;QAClD,MAAM,CAAC,mBAAmB,CAAC,OAAO,EAAE,GAAG,EAAE,GAAE,CAAC,CAAC,CAAC;QAC9C,MAAM,CAAC,mBAAmB,CAAC,QAAQ,EAAE,GAAG,EAAE,CAAC,IAAI,CAAC,cAAc,EAAE,CAAC,CAAC;QAElE,gCAAgC;QAChC,IAAI,IAAI,CAAC,KAAK,EAAE,CAAC;YACf,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;YAC9B,IAAI,CAAC,KAAK,CAAC,QAAQ,CAAC,OAAO,EAAE,CAAC;QAChC,CAAC;QACD,IAAI,CAAC,QAAQ,CAAC,OAAO,EAAE,CAAC;IAC1B,CAAC;wGAtKU,oBAAoB;4FAApB,oBAAoB,mSCbjC,+DACE;;4FDYW,oBAAoB;kBANhC,SAAS;+BACE,kBAAkB,cAGhB,IAAI;+EAGgC,eAAe;sBAA9D,SAAS;uBAAC,iBAAiB,EAAE,EAAE,MAAM,EAAE,IAAI,EAAE;gBACrC,IAAI;sBAAZ,KAAK;gBACG,gBAAgB;sBAAxB,KAAK","sourcesContent":["import { \r\n  Component, ElementRef, AfterViewInit, ViewChild, Input, OnChanges, SimpleChanges, OnDestroy \r\n} from '@angular/core';\r\nimport { HttpClient } from '@angular/common/http';\r\nimport * as THREE from 'three';\r\nimport { STLLoader } from 'three/examples/jsm/loaders/STLLoader.js';\r\n\r\n@Component({\r\n  selector: 'app-three-viewer',\r\n  templateUrl: './three-viewer.component.html',\r\n  styleUrls: ['./three-viewer.component.scss'],\r\n  standalone: true\r\n})\r\nexport class ThreeViewerComponent implements AfterViewInit, OnChanges, OnDestroy {\r\n  @ViewChild('viewerContainer', { static: true }) viewerContainer!: ElementRef;\r\n  @Input() file!: any;\r\n  @Input() manualFileUpload: Blob | null = null;\r\n\r\n  private scene!: THREE.Scene;\r\n  private camera!: THREE.PerspectiveCamera;\r\n  private renderer!: THREE.WebGLRenderer;\r\n  private loader = new STLLoader();\r\n  private model!: THREE.Mesh | null;\r\n\r\n  private isDragging = false;\r\n  private previousMousePosition = { x: 0, y: 0 };\r\n  private zoomSpeed = 1.1;\r\n\r\n  constructor(private http: HttpClient) {}\r\n\r\n  ngAfterViewInit(): void {\r\n    this.initThree();\r\n  }\r\n\r\n  ngOnChanges(changes: SimpleChanges): void {\r\n    if (changes['manualFileUpload']?.currentValue && this.manualFileUpload) {\r\n      this.loadModelFromBlob(this.manualFileUpload);\r\n    }\r\n  }\r\n\r\n  ngOnDestroy(): void {\r\n    this.cleanupThreeJS();\r\n  }\r\n\r\n  private initThree(): void {\r\n    const container = this.viewerContainer.nativeElement;\r\n    this.scene = new THREE.Scene();\r\n    this.scene.background = new THREE.Color(0xf0f0f0);\r\n\r\n    this.camera = new THREE.PerspectiveCamera(60, container.clientWidth / container.clientHeight, 0.1, 1000);\r\n    this.camera.position.set(0, 0, 100);\r\n\r\n    this.renderer = new THREE.WebGLRenderer({ antialias: true });\r\n    this.renderer.setSize(container.clientWidth, container.clientHeight);\r\n    container.appendChild(this.renderer.domElement);\r\n\r\n    const ambientLight = new THREE.AmbientLight(0x404040, 2);\r\n    this.scene.add(ambientLight);\r\n\r\n    const directionalLight = new THREE.DirectionalLight(0xffffff, 1);\r\n    directionalLight.position.set(1, 1, 1).normalize();\r\n    this.scene.add(directionalLight);\r\n\r\n    this.initControls();\r\n    this.animate();\r\n  }\r\n\r\n  private initControls(): void {\r\n    const canvas = this.renderer.domElement;\r\n\r\n    // Mouse Drag for Rotation\r\n    canvas.addEventListener('mousedown', (event) => {\r\n      this.isDragging = true;\r\n      this.previousMousePosition.x = event.clientX;\r\n      this.previousMousePosition.y = event.clientY;\r\n    });\r\n\r\n    canvas.addEventListener('mouseup', () => {\r\n      this.isDragging = false;\r\n    });\r\n\r\n    canvas.addEventListener('mousemove', (event) => {\r\n      if (!this.isDragging) return;\r\n\r\n      const deltaX = event.clientX - this.previousMousePosition.x;\r\n      const deltaY = event.clientY - this.previousMousePosition.y;\r\n\r\n      const rotationSpeed = 0.005;\r\n      this.model?.rotateY(deltaX * rotationSpeed);\r\n      this.model?.rotateX(deltaY * rotationSpeed);\r\n\r\n      this.previousMousePosition.x = event.clientX;\r\n      this.previousMousePosition.y = event.clientY;\r\n    });\r\n\r\n    // Scroll for Zoom\r\n    canvas.addEventListener('wheel', (event) => {\r\n      // event.preventDefault();\r\n      if (event.deltaY < 0) {\r\n        this.camera.position.multiplyScalar(1 / this.zoomSpeed); // Zoom In\r\n      } else {\r\n        this.camera.position.multiplyScalar(this.zoomSpeed); // Zoom Out\r\n      }\r\n      this.camera.updateProjectionMatrix();\r\n    });\r\n\r\n    // Resize Handling\r\n    window.addEventListener('resize', () => this.onWindowResize());\r\n  }\r\n\r\n  private loadModelFromBlob(blob: Blob): void {\r\n    const reader = new FileReader();\r\n    reader.onload = () => {\r\n      const arrayBuffer = reader.result as ArrayBuffer;\r\n      const geometry = this.loader.parse(arrayBuffer);\r\n      geometry.computeVertexNormals();\r\n\r\n      if (this.model) {\r\n        this.scene.remove(this.model);\r\n        this.model.geometry.dispose();\r\n      }\r\n\r\n      const material = new THREE.MeshPhysicalMaterial({\r\n        color: 0xF8F0E3,\r\n        metalness: 0.1,\r\n        roughness: 0.5,\r\n        transmission: 0.5, // Creates slight translucency\r\n        clearcoat: 1.0,    // Adds enamel-like reflection\r\n        clearcoatRoughness: 0.3\r\n      });\r\n      \r\n      this.model = new THREE.Mesh(geometry, material);\r\n      this.scene.add(this.model);\r\n\r\n      this.centerModel();\r\n    };\r\n    reader.readAsArrayBuffer(blob);\r\n  }\r\n\r\n  private centerModel(): void {\r\n    if (!this.model) return;\r\n    const box = new THREE.Box3().setFromObject(this.model);\r\n    const center = new THREE.Vector3();\r\n    box.getCenter(center);\r\n    this.model.position.sub(center);\r\n\r\n    const size = box.getSize(new THREE.Vector3()).length();\r\n    this.camera.position.set(0, 0, size * 2);\r\n    this.camera.updateProjectionMatrix();\r\n    this.onWindowResize();\r\n  }\r\n\r\n  private animate = (): void => {\r\n    requestAnimationFrame(this.animate);\r\n    this.renderer.render(this.scene, this.camera);\r\n  };\r\n\r\n  onWindowResize(): void {\r\n    const container = this.viewerContainer.nativeElement;\r\n    this.camera.aspect = container.clientWidth / container.clientHeight;\r\n    this.camera.updateProjectionMatrix();\r\n    this.renderer.setSize(container.clientWidth, container.clientHeight);\r\n  }\r\n\r\n  private cleanupThreeJS(): void {\r\n    // Remove event listeners\r\n    const canvas = this.renderer.domElement;\r\n    canvas.removeEventListener('mousedown', () => {});\r\n    canvas.removeEventListener('mouseup', () => {});\r\n    canvas.removeEventListener('mousemove', () => {});\r\n    canvas.removeEventListener('wheel', () => {});\r\n    window.removeEventListener('resize', () => this.onWindowResize());\r\n\r\n    // Dispose of Three.js resources\r\n    if (this.model) {\r\n      this.scene.remove(this.model);\r\n      this.model.geometry.dispose();\r\n    }\r\n    this.renderer.dispose();\r\n  }\r\n}\r\n","<div #viewerContainer class=\"viewer-container\"></div>\r\n  "]}
|