@smarterplan/ngx-smarterplan-core 1.2.1 → 1.2.3
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/esm2020/lib/matterport-extensions/hsl-loader/HlsLoader.mjs +68 -0
- package/esm2020/lib/matterport-extensions/nest-thermostat/CanvasRenderer.mjs +2 -3
- package/esm2020/lib/matterport-extensions/security-camera/SecurityCamera.mjs +44 -54
- package/esm2020/lib/matterport-extensions/tv-player/TvPlayer.mjs +98 -0
- package/esm2020/lib/matterport-extensions/video-renderer/VideoRenderer.mjs +64 -0
- package/esm2020/lib/matterport-extensions/view-frustum-mesh/ViewFrustumMesh.mjs +221 -0
- package/esm2020/lib/services/matterport.service.mjs +70 -9
- package/esm2020/lib/services/viewer.service.mjs +4 -2
- package/esm2020/lib/types.service.mjs +1 -1
- package/fesm2015/smarterplan-ngx-smarterplan-core.mjs +439 -73
- package/fesm2015/smarterplan-ngx-smarterplan-core.mjs.map +1 -1
- package/fesm2020/smarterplan-ngx-smarterplan-core.mjs +439 -73
- package/fesm2020/smarterplan-ngx-smarterplan-core.mjs.map +1 -1
- package/lib/matterport-extensions/hsl-loader/HlsLoader.d.ts +26 -0
- package/lib/matterport-extensions/hsl-loader/HlsLoader.d.ts.map +1 -0
- package/lib/matterport-extensions/nest-thermostat/CanvasRenderer.d.ts +1 -2
- package/lib/matterport-extensions/nest-thermostat/CanvasRenderer.d.ts.map +1 -1
- package/lib/matterport-extensions/security-camera/SecurityCamera.d.ts +7 -6
- package/lib/matterport-extensions/security-camera/SecurityCamera.d.ts.map +1 -1
- package/lib/matterport-extensions/tv-player/TvPlayer.d.ts +26 -0
- package/lib/matterport-extensions/tv-player/TvPlayer.d.ts.map +1 -0
- package/lib/matterport-extensions/video-renderer/VideoRenderer.d.ts +26 -0
- package/lib/matterport-extensions/video-renderer/VideoRenderer.d.ts.map +1 -0
- package/lib/matterport-extensions/view-frustum-mesh/ViewFrustumMesh.d.ts +43 -0
- package/lib/matterport-extensions/view-frustum-mesh/ViewFrustumMesh.d.ts.map +1 -0
- package/lib/services/matterport.service.d.ts +2 -0
- package/lib/services/matterport.service.d.ts.map +1 -1
- package/lib/services/viewer.service.d.ts.map +1 -1
- package/lib/types.service.d.ts +1 -0
- package/lib/types.service.d.ts.map +1 -1
- package/package.json +1 -1
|
@@ -0,0 +1,68 @@
|
|
|
1
|
+
import { SceneComponent } from './../scene-component/SceneComponent';
|
|
2
|
+
// @ts-ignore
|
|
3
|
+
import Hls from 'hls.js';
|
|
4
|
+
export class HlsLoader extends SceneComponent {
|
|
5
|
+
constructor() {
|
|
6
|
+
super(...arguments);
|
|
7
|
+
this.video = null;
|
|
8
|
+
this.texture = null;
|
|
9
|
+
this.hls = null;
|
|
10
|
+
this.inputs = {
|
|
11
|
+
src: './../assets/video/121222-04.mp4',
|
|
12
|
+
enabled: true,
|
|
13
|
+
};
|
|
14
|
+
this.outputs = {
|
|
15
|
+
video: null,
|
|
16
|
+
aspect: 1,
|
|
17
|
+
};
|
|
18
|
+
}
|
|
19
|
+
onInit(videoTexture) {
|
|
20
|
+
this.outputs.video = videoTexture;
|
|
21
|
+
this.outputs.aspect = 720 / 480;
|
|
22
|
+
if (this.inputs.enabled) {
|
|
23
|
+
this.setupStream();
|
|
24
|
+
}
|
|
25
|
+
}
|
|
26
|
+
onInputsUpdated() {
|
|
27
|
+
this.setupStream();
|
|
28
|
+
}
|
|
29
|
+
onDestroy() {
|
|
30
|
+
this.releaseResources();
|
|
31
|
+
}
|
|
32
|
+
createVideoElement() {
|
|
33
|
+
const video = document.createElement('video');
|
|
34
|
+
video.crossOrigin = 'anonymous';
|
|
35
|
+
video.setAttribute('height', '480');
|
|
36
|
+
video.setAttribute('width', '720');
|
|
37
|
+
video.volume = 0.1;
|
|
38
|
+
return video;
|
|
39
|
+
}
|
|
40
|
+
setupStream() {
|
|
41
|
+
this.releaseResources();
|
|
42
|
+
if (this.inputs.enabled) {
|
|
43
|
+
this.video = this.createVideoElement();
|
|
44
|
+
this.hls = new Hls();
|
|
45
|
+
this.hls.loadSource(this.inputs.src);
|
|
46
|
+
this.hls.attachMedia(this.video);
|
|
47
|
+
this.hls.on(Hls.Events.MANIFEST_PARSED, () => {
|
|
48
|
+
this.outputs.video = this.video;
|
|
49
|
+
});
|
|
50
|
+
}
|
|
51
|
+
}
|
|
52
|
+
releaseResources() {
|
|
53
|
+
if (this.hls) {
|
|
54
|
+
this.hls.destroy();
|
|
55
|
+
this.hls = null;
|
|
56
|
+
}
|
|
57
|
+
if (this.texture) {
|
|
58
|
+
this.texture.dispose();
|
|
59
|
+
this.texture = null;
|
|
60
|
+
this.outputs.video = null;
|
|
61
|
+
}
|
|
62
|
+
}
|
|
63
|
+
}
|
|
64
|
+
export const hlsLoaderType = 'mp.hlsLoader';
|
|
65
|
+
export function makeHlsLoader() {
|
|
66
|
+
return new HlsLoader();
|
|
67
|
+
}
|
|
68
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiSGxzTG9hZGVyLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvbmd4LXNtYXJ0ZXJwbGFuLWNvcmUvc3JjL2xpYi9tYXR0ZXJwb3J0LWV4dGVuc2lvbnMvaHNsLWxvYWRlci9IbHNMb2FkZXIudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLGNBQWMsRUFBbUIsTUFBTSxxQ0FBcUMsQ0FBQztBQUV0RixhQUFhO0FBQ2IsT0FBTyxHQUFHLE1BQU0sUUFBUSxDQUFDO0FBWXpCLE1BQU0sT0FBTyxTQUFVLFNBQVEsY0FBYztJQUE3Qzs7UUFDVSxVQUFLLEdBQTRCLElBQUksQ0FBQztRQUN0QyxZQUFPLEdBQXdCLElBQUksQ0FBQztRQUNwQyxRQUFHLEdBQWUsSUFBSSxDQUFDO1FBRXRCLFdBQU0sR0FBVztZQUN4QixHQUFHLEVBQUUsaUNBQWlDO1lBQ3RDLE9BQU8sRUFBRSxJQUFJO1NBQ2QsQ0FBQTtRQUVRLFlBQU8sR0FBRztZQUNqQixLQUFLLEVBQUUsSUFBSTtZQUNYLE1BQU0sRUFBRSxDQUFDO1NBQ0MsQ0FBQztJQXNEZixDQUFDO0lBcERVLE1BQU0sQ0FBQyxZQUEyQjtRQUN6QyxJQUFJLENBQUMsT0FBTyxDQUFDLEtBQUssR0FBRyxZQUFZLENBQUM7UUFDbEMsSUFBSSxDQUFDLE9BQU8sQ0FBQyxNQUFNLEdBQUcsR0FBRyxHQUFDLEdBQUcsQ0FBQztRQUM5QixJQUFJLElBQUksQ0FBQyxNQUFNLENBQUMsT0FBTyxFQUFFO1lBQ3ZCLElBQUksQ0FBQyxXQUFXLEVBQUUsQ0FBQztTQUNwQjtJQUNILENBQUM7SUFFUSxlQUFlO1FBRXRCLElBQUksQ0FBQyxXQUFXLEVBQUUsQ0FBQztJQUNyQixDQUFDO0lBRVEsU0FBUztRQUNoQixJQUFJLENBQUMsZ0JBQWdCLEVBQUUsQ0FBQztJQUMxQixDQUFDO0lBRU8sa0JBQWtCO1FBQ3hCLE1BQU0sS0FBSyxHQUFHLFFBQVEsQ0FBQyxhQUFhLENBQUMsT0FBTyxDQUFDLENBQUM7UUFDOUMsS0FBSyxDQUFDLFdBQVcsR0FBRyxXQUFXLENBQUM7UUFDaEMsS0FBSyxDQUFDLFlBQVksQ0FBQyxRQUFRLEVBQUUsS0FBSyxDQUFDLENBQUM7UUFDcEMsS0FBSyxDQUFDLFlBQVksQ0FBQyxPQUFPLEVBQUUsS0FBSyxDQUFDLENBQUM7UUFDbkMsS0FBSyxDQUFDLE1BQU0sR0FBRyxHQUFHLENBQUM7UUFDbkIsT0FBTyxLQUFLLENBQUM7SUFDZixDQUFDO0lBRU8sV0FBVztRQUNqQixJQUFJLENBQUMsZ0JBQWdCLEVBQUUsQ0FBQztRQUV4QixJQUFJLElBQUksQ0FBQyxNQUFNLENBQUMsT0FBTyxFQUFFO1lBQ3ZCLElBQUksQ0FBQyxLQUFLLEdBQUcsSUFBSSxDQUFDLGtCQUFrQixFQUFFLENBQUM7WUFDdkMsSUFBSSxDQUFDLEdBQUcsR0FBRyxJQUFJLEdBQUcsRUFBRSxDQUFDO1lBQ3JCLElBQUksQ0FBQyxHQUFHLENBQUMsVUFBVSxDQUFDLElBQUksQ0FBQyxNQUFNLENBQUMsR0FBRyxDQUFDLENBQUM7WUFDckMsSUFBSSxDQUFDLEdBQUcsQ0FBQyxXQUFXLENBQUMsSUFBSSxDQUFDLEtBQUssQ0FBQyxDQUFDO1lBQ2pDLElBQUksQ0FBQyxHQUFHLENBQUMsRUFBRSxDQUFDLEdBQUcsQ0FBQyxNQUFNLENBQUMsZUFBZSxFQUFFLEdBQUcsRUFBRTtnQkFDM0MsSUFBSSxDQUFDLE9BQU8sQ0FBQyxLQUFLLEdBQUcsSUFBSSxDQUFDLEtBQUssQ0FBQztZQUNsQyxDQUFDLENBQUMsQ0FBQztTQUNKO0lBQ0gsQ0FBQztJQUVPLGdCQUFnQjtRQUN0QixJQUFJLElBQUksQ0FBQyxHQUFHLEVBQUU7WUFDWixJQUFJLENBQUMsR0FBRyxDQUFDLE9BQU8sRUFBRSxDQUFDO1lBQ25CLElBQUksQ0FBQyxHQUFHLEdBQUcsSUFBSSxDQUFDO1NBQ2pCO1FBRUQsSUFBSSxJQUFJLENBQUMsT0FBTyxFQUFFO1lBQ2hCLElBQUksQ0FBQyxPQUFPLENBQUMsT0FBTyxFQUFFLENBQUM7WUFDdkIsSUFBSSxDQUFDLE9BQU8sR0FBRyxJQUFJLENBQUM7WUFDcEIsSUFBSSxDQUFDLE9BQU8sQ0FBQyxLQUFLLEdBQUcsSUFBSSxDQUFDO1NBQzNCO0lBQ0gsQ0FBQztDQUNGO0FBRUQsTUFBTSxDQUFDLE1BQU0sYUFBYSxHQUFHLGNBQWMsQ0FBQztBQUM1QyxNQUFNLFVBQVUsYUFBYTtJQUMzQixPQUFPLElBQUksU0FBUyxFQUFFLENBQUM7QUFDekIsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IFNjZW5lQ29tcG9uZW50LCBDb21wb25lbnRPdXRwdXQgfSBmcm9tICcuLy4uL3NjZW5lLWNvbXBvbmVudC9TY2VuZUNvbXBvbmVudCc7XHJcbmltcG9ydCB7IFZpZGVvVGV4dHVyZSB9IGZyb20gJ3RocmVlJztcclxuLy8gQHRzLWlnbm9yZVxyXG5pbXBvcnQgSGxzIGZyb20gJ2hscy5qcyc7XHJcblxyXG50eXBlIElucHV0cyA9IHtcclxuICBzcmM6IHN0cmluZztcclxuICBlbmFibGVkOiBib29sZWFuO1xyXG59O1xyXG5cclxudHlwZSBPdXRwdXRzID0ge1xyXG4gIHZpZGVvOiBIVE1MVmlkZW9FbGVtZW50IHwgbnVsbDtcclxuICBhc3BlY3Q6IG51bWJlcjtcclxufSAmIENvbXBvbmVudE91dHB1dDtcclxuXHJcbmV4cG9ydCBjbGFzcyBIbHNMb2FkZXIgZXh0ZW5kcyBTY2VuZUNvbXBvbmVudCB7XHJcbiAgcHJpdmF0ZSB2aWRlbzogSFRNTFZpZGVvRWxlbWVudCB8IG51bGwgPSBudWxsO1xyXG4gIHByaXZhdGUgdGV4dHVyZTogVmlkZW9UZXh0dXJlIHwgbnVsbCA9IG51bGw7XHJcbiAgcHJpdmF0ZSBobHM6IEhscyB8IG51bGwgPSBudWxsO1xyXG5cclxuICBvdmVycmlkZSBpbnB1dHM6IElucHV0cyA9IHtcclxuICAgIHNyYzogJy4vLi4vYXNzZXRzL3ZpZGVvLzEyMTIyMi0wNC5tcDQnLFxyXG4gICAgZW5hYmxlZDogdHJ1ZSxcclxuICB9XHJcblxyXG4gIG92ZXJyaWRlIG91dHB1dHMgPSB7XHJcbiAgICB2aWRlbzogbnVsbCxcclxuICAgIGFzcGVjdDogMSxcclxuICB9IGFzIE91dHB1dHM7XHJcblxyXG4gIG92ZXJyaWRlIG9uSW5pdCh2aWRlb1RleHR1cmU/OiBWaWRlb1RleHR1cmUpIHtcclxuICAgIHRoaXMub3V0cHV0cy52aWRlbyA9IHZpZGVvVGV4dHVyZTtcclxuICAgIHRoaXMub3V0cHV0cy5hc3BlY3QgPSA3MjAvNDgwO1xyXG4gICAgaWYgKHRoaXMuaW5wdXRzLmVuYWJsZWQpIHtcclxuICAgICAgdGhpcy5zZXR1cFN0cmVhbSgpO1xyXG4gICAgfVxyXG4gIH1cclxuXHJcbiAgb3ZlcnJpZGUgb25JbnB1dHNVcGRhdGVkKCkge1xyXG5cclxuICAgIHRoaXMuc2V0dXBTdHJlYW0oKTtcclxuICB9XHJcblxyXG4gIG92ZXJyaWRlIG9uRGVzdHJveSgpIHtcclxuICAgIHRoaXMucmVsZWFzZVJlc291cmNlcygpO1xyXG4gIH1cclxuXHJcbiAgcHJpdmF0ZSBjcmVhdGVWaWRlb0VsZW1lbnQoKSB7XHJcbiAgICBjb25zdCB2aWRlbyA9IGRvY3VtZW50LmNyZWF0ZUVsZW1lbnQoJ3ZpZGVvJyk7XHJcbiAgICB2aWRlby5jcm9zc09yaWdpbiA9ICdhbm9ueW1vdXMnO1xyXG4gICAgdmlkZW8uc2V0QXR0cmlidXRlKCdoZWlnaHQnLCAnNDgwJyk7XHJcbiAgICB2aWRlby5zZXRBdHRyaWJ1dGUoJ3dpZHRoJywgJzcyMCcpO1xyXG4gICAgdmlkZW8udm9sdW1lID0gMC4xO1xyXG4gICAgcmV0dXJuIHZpZGVvO1xyXG4gIH1cclxuXHJcbiAgcHJpdmF0ZSBzZXR1cFN0cmVhbSgpIHtcclxuICAgIHRoaXMucmVsZWFzZVJlc291cmNlcygpO1xyXG5cclxuICAgIGlmICh0aGlzLmlucHV0cy5lbmFibGVkKSB7XHJcbiAgICAgIHRoaXMudmlkZW8gPSB0aGlzLmNyZWF0ZVZpZGVvRWxlbWVudCgpO1xyXG4gICAgICB0aGlzLmhscyA9IG5ldyBIbHMoKTtcclxuICAgICAgdGhpcy5obHMubG9hZFNvdXJjZSh0aGlzLmlucHV0cy5zcmMpO1xyXG4gICAgICB0aGlzLmhscy5hdHRhY2hNZWRpYSh0aGlzLnZpZGVvKTtcclxuICAgICAgdGhpcy5obHMub24oSGxzLkV2ZW50cy5NQU5JRkVTVF9QQVJTRUQsICgpID0+IHtcclxuICAgICAgICB0aGlzLm91dHB1dHMudmlkZW8gPSB0aGlzLnZpZGVvO1xyXG4gICAgICB9KTtcclxuICAgIH1cclxuICB9XHJcblxyXG4gIHByaXZhdGUgcmVsZWFzZVJlc291cmNlcygpIHtcclxuICAgIGlmICh0aGlzLmhscykge1xyXG4gICAgICB0aGlzLmhscy5kZXN0cm95KCk7XHJcbiAgICAgIHRoaXMuaGxzID0gbnVsbDtcclxuICAgIH1cclxuXHJcbiAgICBpZiAodGhpcy50ZXh0dXJlKSB7XHJcbiAgICAgIHRoaXMudGV4dHVyZS5kaXNwb3NlKCk7XHJcbiAgICAgIHRoaXMudGV4dHVyZSA9IG51bGw7XHJcbiAgICAgIHRoaXMub3V0cHV0cy52aWRlbyA9IG51bGw7XHJcbiAgICB9XHJcbiAgfVxyXG59XHJcblxyXG5leHBvcnQgY29uc3QgaGxzTG9hZGVyVHlwZSA9ICdtcC5obHNMb2FkZXInO1xyXG5leHBvcnQgZnVuY3Rpb24gbWFrZUhsc0xvYWRlcigpIHtcclxuICByZXR1cm4gbmV3IEhsc0xvYWRlcigpO1xyXG59XHJcbiJdfQ==
|
|
@@ -14,12 +14,11 @@ export class CanvasRenderer extends SceneComponent {
|
|
|
14
14
|
repaint: true,
|
|
15
15
|
};
|
|
16
16
|
}
|
|
17
|
-
onInit(
|
|
17
|
+
onInit() {
|
|
18
18
|
// set up canvas 2d context
|
|
19
19
|
this.canvas = document.createElement('canvas');
|
|
20
20
|
this.renderContext2D = this.canvas.getContext('2d');
|
|
21
21
|
this.outputs.texture = new CanvasTexture(this.canvas);
|
|
22
|
-
// this.inputs.painter = ci; // FOR TV ONLY
|
|
23
22
|
this.resize(this.inputs.textureRes);
|
|
24
23
|
this.repaint();
|
|
25
24
|
}
|
|
@@ -59,4 +58,4 @@ export const canvasRendererType = 'mp.canvasRenderer';
|
|
|
59
58
|
export function makeCanvasRenderer() {
|
|
60
59
|
return new CanvasRenderer();
|
|
61
60
|
}
|
|
62
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
61
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiQ2FudmFzUmVuZGVyZXIuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9uZ3gtc21hcnRlcnBsYW4tY29yZS9zcmMvbGliL21hdHRlcnBvcnQtZXh0ZW5zaW9ucy9uZXN0LXRoZXJtb3N0YXQvQ2FudmFzUmVuZGVyZXIudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLGNBQWMsRUFBbUIsTUFBTSxtQ0FBbUMsQ0FBQztBQUVwRixPQUFPLEVBQUUsYUFBYSxFQUFFLE1BQU0sT0FBTyxDQUFDO0FBcUJ0QyxNQUFNLE9BQU8sY0FBZSxTQUFRLGNBQWM7SUFBbEQ7O1FBSVcsV0FBTSxHQUFXO1lBQ3hCLE9BQU8sRUFBRSxJQUFJO1lBQ2IsVUFBVSxFQUFFLEVBQUUsQ0FBQyxFQUFFLElBQUksRUFBRSxDQUFDLEVBQUUsSUFBSSxFQUFFLEVBQUUscUJBQXFCO1NBQ3hELENBQUE7UUFFUSxZQUFPLEdBQUc7WUFDakIsT0FBTyxFQUFFLElBQUk7U0FDSCxDQUFDO1FBRUosV0FBTSxHQUFHO1lBQ2hCLE9BQU8sRUFBRSxJQUFJO1NBQ0osQ0FBQztJQW1EZCxDQUFDO0lBakRVLE1BQU07UUFFYiwyQkFBMkI7UUFDM0IsSUFBSSxDQUFDLE1BQU0sR0FBRyxRQUFRLENBQUMsYUFBYSxDQUFDLFFBQVEsQ0FBQyxDQUFDO1FBQy9DLElBQUksQ0FBQyxlQUFlLEdBQUcsSUFBSSxDQUFDLE1BQU0sQ0FBQyxVQUFVLENBQUMsSUFBSSxDQUFDLENBQUM7UUFDcEQsSUFBSSxDQUFDLE9BQU8sQ0FBQyxPQUFPLEdBQUcsSUFBSSxhQUFhLENBQUMsSUFBSSxDQUFDLE1BQU0sQ0FBQyxDQUFDO1FBRXRELElBQUksQ0FBQyxNQUFNLENBQUMsSUFBSSxDQUFDLE1BQU0sQ0FBQyxVQUFVLENBQUMsQ0FBQztRQUNwQyxJQUFJLENBQUMsT0FBTyxFQUFFLENBQUM7SUFDakIsQ0FBQztJQUVELDhCQUE4QixDQUFDLEVBQUU7UUFDL0IsSUFBSSxDQUFDLE1BQU0sQ0FBQyxPQUFPLEdBQUcsRUFBRSxDQUFDO0lBQzNCLENBQUM7SUFFUSxlQUFlLENBQUMsU0FBaUI7UUFDeEMsSUFBSSxTQUFTLENBQUMsVUFBVSxDQUFDLENBQUMsS0FBSyxJQUFJLENBQUMsTUFBTSxDQUFDLFVBQVUsQ0FBQyxDQUFDO1lBQ25ELFNBQVMsQ0FBQyxVQUFVLENBQUMsQ0FBQyxLQUFLLElBQUksQ0FBQyxNQUFNLENBQUMsVUFBVSxDQUFDLENBQUMsRUFBRTtZQUN2RCxJQUFJLENBQUMsTUFBTSxDQUFDLElBQUksQ0FBQyxNQUFNLENBQUMsVUFBVSxDQUFDLENBQUM7U0FDckM7UUFFRCxJQUFJLFNBQVMsQ0FBQyxPQUFPLEtBQUssSUFBSSxDQUFDLE1BQU0sQ0FBQyxPQUFPLEVBQUU7WUFDN0MsSUFBSSxDQUFDLE9BQU8sRUFBRSxDQUFDO1NBQ2hCO0lBQ0gsQ0FBQztJQUVRLE9BQU8sQ0FBQyxTQUFpQixFQUFFLFVBQW1CO1FBQ3JELElBQUksU0FBUyxLQUFLLFNBQVMsRUFBRTtZQUMzQixJQUFJLENBQUMsT0FBTyxFQUFFLENBQUM7U0FDaEI7SUFDSCxDQUFDO0lBRVEsU0FBUztRQUNoQixJQUFJLENBQUMsT0FBTyxDQUFDLE9BQU8sQ0FBQyxPQUFPLEVBQUUsQ0FBQztRQUMvQixJQUFJLENBQUMsT0FBTyxDQUFDLE9BQU8sR0FBRyxJQUFJLENBQUM7SUFDOUIsQ0FBQztJQUVPLE1BQU0sQ0FBQyxJQUFVO1FBQ3ZCLElBQUksQ0FBQyxNQUFNLENBQUMsS0FBSyxHQUFHLElBQUksQ0FBQyxDQUFDLENBQUM7UUFDM0IsSUFBSSxDQUFDLE1BQU0sQ0FBQyxNQUFNLEdBQUcsSUFBSSxDQUFDLENBQUMsQ0FBQztJQUM5QixDQUFDO0lBRU0sT0FBTztRQUNaLElBQUksSUFBSSxDQUFDLE1BQU0sQ0FBQyxPQUFPLEVBQUU7WUFDdkIsSUFBSSxDQUFDLE1BQU0sQ0FBQyxPQUFPLENBQUMsS0FBSyxDQUFDLElBQUksQ0FBQyxlQUFlLEVBQUUsSUFBSSxDQUFDLE1BQU0sQ0FBQyxVQUFVLENBQUMsQ0FBQztZQUN4RSxJQUFJLENBQUMsT0FBTyxDQUFDLE9BQU8sQ0FBQyxXQUFXLEdBQUcsSUFBSSxDQUFDO1NBQ3pDO0lBQ0gsQ0FBQztDQUVGO0FBT0QsTUFBTSxDQUFDLE1BQU0sa0JBQWtCLEdBQUcsbUJBQW1CLENBQUM7QUFDdEQsTUFBTSxVQUFVLGtCQUFrQjtJQUNoQyxPQUFPLElBQUksY0FBYyxFQUFFLENBQUM7QUFDOUIsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IFNjZW5lQ29tcG9uZW50LCBDb21wb25lbnRPdXRwdXQgfSBmcm9tICcuLi9zY2VuZS1jb21wb25lbnQvU2NlbmVDb21wb25lbnQnO1xuaW1wb3J0IHR5cGUgeyBUZXh0dXJlIH0gZnJvbSAndGhyZWUnO1xuaW1wb3J0IHsgQ2FudmFzVGV4dHVyZSB9IGZyb20gJ3RocmVlJztcbmltcG9ydCB7IFNpemUgfSBmcm9tICcuL1BsYW5lUmVuZGVyZXInO1xuaW1wb3J0IHsgQ2FudmFzSW1hZ2UgfSBmcm9tICcuL0NhbnZhc0ltYWdlJztcblxuZXhwb3J0IGludGVyZmFjZSBJUGFpbnRlcjJkIHtcbiAgcGFpbnQoY29udGV4dDJkOiBDYW52YXNSZW5kZXJpbmdDb250ZXh0MkQsIHNpemU6IFNpemUpOiB2b2lkO1xufVxuXG50eXBlIElucHV0cyA9IHtcbiAgcGFpbnRlcjogSVBhaW50ZXIyZCB8IG51bGw7XG4gIHRleHR1cmVSZXM6IFNpemU7XG59XG5cbnR5cGUgT3V0cHV0cyA9IHtcbiAgdGV4dHVyZTogVGV4dHVyZSB8IG51bGw7XG59ICYgQ29tcG9uZW50T3V0cHV0O1xuXG50eXBlIEV2ZW50cyA9IHtcbiAgcmVwYWludDogYm9vbGVhbjtcbn07XG5cbmV4cG9ydCBjbGFzcyBDYW52YXNSZW5kZXJlciBleHRlbmRzIFNjZW5lQ29tcG9uZW50IHtcbiAgcHJpdmF0ZSBjYW52YXM6IEhUTUxDYW52YXNFbGVtZW50O1xuICBwcml2YXRlIHJlbmRlckNvbnRleHQyRDogQ2FudmFzUmVuZGVyaW5nQ29udGV4dDJEO1xuXG4gIG92ZXJyaWRlIGlucHV0czogSW5wdXRzID0ge1xuICAgIHBhaW50ZXI6IG51bGwsXG4gICAgdGV4dHVyZVJlczogeyB3OiAxMDI0LCBoOiAxMDI0IH0sIC8vIGRpbSBmb3IgdGhlcm1vc3RhdFxuICB9XG5cbiAgb3ZlcnJpZGUgb3V0cHV0cyA9IHtcbiAgICB0ZXh0dXJlOiBudWxsLFxuICB9IGFzIE91dHB1dHM7XG5cbiAgb3ZlcnJpZGUgZXZlbnRzID0ge1xuICAgIHJlcGFpbnQ6IHRydWUsXG4gIH0gYXMgRXZlbnRzO1xuXG4gIG92ZXJyaWRlIG9uSW5pdCgpIHtcblxuICAgIC8vIHNldCB1cCBjYW52YXMgMmQgY29udGV4dFxuICAgIHRoaXMuY2FudmFzID0gZG9jdW1lbnQuY3JlYXRlRWxlbWVudCgnY2FudmFzJyk7XG4gICAgdGhpcy5yZW5kZXJDb250ZXh0MkQgPSB0aGlzLmNhbnZhcy5nZXRDb250ZXh0KCcyZCcpO1xuICAgIHRoaXMub3V0cHV0cy50ZXh0dXJlID0gbmV3IENhbnZhc1RleHR1cmUodGhpcy5jYW52YXMpO1xuXG4gICAgdGhpcy5yZXNpemUodGhpcy5pbnB1dHMudGV4dHVyZVJlcyk7XG4gICAgdGhpcy5yZXBhaW50KCk7XG4gIH1cblxuICBzZXRDYW52YXNOZXN0VGhlcm1vc3RhdFBhaW50ZXIoc2Mpe1xuICAgIHRoaXMuaW5wdXRzLnBhaW50ZXIgPSBzYztcbiAgfVxuXG4gIG92ZXJyaWRlIG9uSW5wdXRzVXBkYXRlZChvbGRJbnB1dHM6IElucHV0cykge1xuICAgIGlmIChvbGRJbnB1dHMudGV4dHVyZVJlcy53ICE9PSB0aGlzLmlucHV0cy50ZXh0dXJlUmVzLncgfHxcbiAgICAgICAgb2xkSW5wdXRzLnRleHR1cmVSZXMuaCAhPT0gdGhpcy5pbnB1dHMudGV4dHVyZVJlcy5oKSB7XG4gICAgICB0aGlzLnJlc2l6ZSh0aGlzLmlucHV0cy50ZXh0dXJlUmVzKTtcbiAgICB9XG5cbiAgICBpZiAob2xkSW5wdXRzLnBhaW50ZXIgIT09IHRoaXMuaW5wdXRzLnBhaW50ZXIpIHtcbiAgICAgIHRoaXMucmVwYWludCgpO1xuICAgIH1cbiAgfVxuXG4gIG92ZXJyaWRlIG9uRXZlbnQoZXZlbnRUeXBlOiBzdHJpbmcsIF9ldmVudERhdGE6IHVua25vd24pIHtcbiAgICBpZiAoZXZlbnRUeXBlID09PSAncmVwYWludCcpIHtcbiAgICAgIHRoaXMucmVwYWludCgpO1xuICAgIH1cbiAgfVxuXG4gIG92ZXJyaWRlIG9uRGVzdHJveSgpIHtcbiAgICB0aGlzLm91dHB1dHMudGV4dHVyZS5kaXNwb3NlKCk7XG4gICAgdGhpcy5vdXRwdXRzLnRleHR1cmUgPSBudWxsO1xuICB9XG5cbiAgcHJpdmF0ZSByZXNpemUoc2l6ZTogU2l6ZSkge1xuICAgIHRoaXMuY2FudmFzLndpZHRoID0gc2l6ZS53O1xuICAgIHRoaXMuY2FudmFzLmhlaWdodCA9IHNpemUuaDtcbiAgfVxuXG4gIHB1YmxpYyByZXBhaW50KCkge1xuICAgIGlmICh0aGlzLmlucHV0cy5wYWludGVyKSB7XG4gICAgICB0aGlzLmlucHV0cy5wYWludGVyLnBhaW50KHRoaXMucmVuZGVyQ29udGV4dDJELCB0aGlzLmlucHV0cy50ZXh0dXJlUmVzKTtcbiAgICAgIHRoaXMub3V0cHV0cy50ZXh0dXJlLm5lZWRzVXBkYXRlID0gdHJ1ZTtcbiAgICB9XG4gIH1cblxufVxuXG5leHBvcnQgaW50ZXJmYWNlIElDYW52YXNSZW5kZXJlciBleHRlbmRzIFNjZW5lQ29tcG9uZW50e1xuICBpbnB1dHM6IElucHV0cztcbiAgb3V0cHV0czogT3V0cHV0cztcbn1cblxuZXhwb3J0IGNvbnN0IGNhbnZhc1JlbmRlcmVyVHlwZSA9ICdtcC5jYW52YXNSZW5kZXJlcic7XG5leHBvcnQgZnVuY3Rpb24gbWFrZUNhbnZhc1JlbmRlcmVyKCkge1xuICByZXR1cm4gbmV3IENhbnZhc1JlbmRlcmVyKCk7XG59XG4iXX0=
|
|
@@ -1,10 +1,8 @@
|
|
|
1
|
-
import {
|
|
2
|
-
import * as BufferGeometryUtils from 'three/examples/jsm/utils/BufferGeometryUtils.js';
|
|
1
|
+
import { Color, Matrix4, Vector3 } from 'three';
|
|
3
2
|
import { SceneComponent } from '../scene-component/SceneComponent';
|
|
4
3
|
export class SecurityCamera extends SceneComponent {
|
|
5
4
|
constructor() {
|
|
6
5
|
super(...arguments);
|
|
7
|
-
this.rootScene = null;
|
|
8
6
|
this.root = null;
|
|
9
7
|
this.pivot = null;
|
|
10
8
|
this.box = null;
|
|
@@ -28,32 +26,20 @@ export class SecurityCamera extends SceneComponent {
|
|
|
28
26
|
};
|
|
29
27
|
this.outputs = {};
|
|
30
28
|
}
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
this.
|
|
34
|
-
this.
|
|
35
|
-
}
|
|
36
|
-
setRootScene(rootScene) {
|
|
37
|
-
this.rootScene = rootScene;
|
|
38
|
-
}
|
|
39
|
-
onInit(modelNode) {
|
|
40
|
-
if (!this.context) {
|
|
41
|
-
this.context = {
|
|
42
|
-
root: modelNode
|
|
43
|
-
};
|
|
44
|
-
}
|
|
45
|
-
this.root = new Object3D();
|
|
46
|
-
this.pivot = new Object3D();
|
|
29
|
+
onInit() {
|
|
30
|
+
const THREE = this.context.three;
|
|
31
|
+
this.root = new THREE.Object3D();
|
|
32
|
+
this.pivot = new THREE.Object3D();
|
|
47
33
|
this.root.add(this.pivot);
|
|
48
34
|
this.outputs.objectRoot = this.root;
|
|
49
35
|
this.root.position.set(this.inputs.localPosition.x, this.inputs.localPosition.y, this.inputs.localPosition.z);
|
|
50
|
-
const euler = new Euler(this.inputs.localRotation.x * Math.PI / 180, this.inputs.localRotation.y * Math.PI / 180, this.inputs.localRotation.z * Math.PI / 180, 'YXZ');
|
|
36
|
+
const euler = new THREE.Euler(this.inputs.localRotation.x * Math.PI / 180, this.inputs.localRotation.y * Math.PI / 180, this.inputs.localRotation.z * Math.PI / 180, 'YXZ');
|
|
51
37
|
this.pivot.quaternion.setFromEuler(euler);
|
|
52
38
|
const aspect = this.inputs.aspect;
|
|
53
39
|
const DEG2RAD = Math.PI / 180;
|
|
54
40
|
const RAD2DEG = 1 / DEG2RAD;
|
|
55
41
|
const verticalFOV = 2 * Math.atan(1 / aspect * Math.tan(0.5 * this.inputs.horizontalFOV * DEG2RAD)) * RAD2DEG;
|
|
56
|
-
this.projector = new PerspectiveCamera(verticalFOV, aspect, this.inputs.nearPlane, this.inputs.farPlane);
|
|
42
|
+
this.projector = new THREE.PerspectiveCamera(verticalFOV, aspect, this.inputs.nearPlane, this.inputs.farPlane);
|
|
57
43
|
// orientation of the projector is handled by the pivot
|
|
58
44
|
this.pivot.add(this.projector);
|
|
59
45
|
this.makeFrustumVisuals();
|
|
@@ -62,8 +48,7 @@ export class SecurityCamera extends SceneComponent {
|
|
|
62
48
|
this.makeHighlight();
|
|
63
49
|
// attach highighting to the mesh it represents
|
|
64
50
|
// TODO (scene query): this is very brittle but is the only way to attach something to a room of the model
|
|
65
|
-
|
|
66
|
-
this.rootScene.add(this.highlight);
|
|
51
|
+
this.outputs.objectRoot.parent.parent.add(this.highlight);
|
|
67
52
|
// create the animation mixer to rotate the camera
|
|
68
53
|
this.makeAnimation();
|
|
69
54
|
}
|
|
@@ -88,6 +73,7 @@ export class SecurityCamera extends SceneComponent {
|
|
|
88
73
|
}
|
|
89
74
|
}
|
|
90
75
|
makeFrustumVisuals() {
|
|
76
|
+
const THREE = this.context.three;
|
|
91
77
|
if (this.box) {
|
|
92
78
|
this.root.remove(this.box);
|
|
93
79
|
this.box.material.dispose();
|
|
@@ -106,11 +92,11 @@ export class SecurityCamera extends SceneComponent {
|
|
|
106
92
|
const r = thickness / 2;
|
|
107
93
|
const geoms = [];
|
|
108
94
|
for (let i = 0; i < count * 3 - 1; i += 6) {
|
|
109
|
-
const a = new Vector3(array[i], array[i + 1], array[i + 2]);
|
|
110
|
-
const b = new Vector3(array[i + 3], array[i + 4], array[i + 5]);
|
|
111
|
-
const vec = new Vector3().subVectors(b, a);
|
|
95
|
+
const a = new THREE.Vector3(array[i], array[i + 1], array[i + 2]);
|
|
96
|
+
const b = new THREE.Vector3(array[i + 3], array[i + 4], array[i + 5]);
|
|
97
|
+
const vec = new THREE.Vector3().subVectors(b, a);
|
|
112
98
|
const len = vec.length();
|
|
113
|
-
const geom = new CylinderBufferGeometry(r, r, len, 8);
|
|
99
|
+
const geom = new THREE.CylinderBufferGeometry(r, r, len, 8);
|
|
114
100
|
geom.translate(0, len / 2, 0);
|
|
115
101
|
geom.rotateX(Math.PI / 2);
|
|
116
102
|
geom.lookAt(vec);
|
|
@@ -118,10 +104,10 @@ export class SecurityCamera extends SceneComponent {
|
|
|
118
104
|
geoms.push(geom);
|
|
119
105
|
}
|
|
120
106
|
// API to be updated in bundle - JSSDK-1222
|
|
121
|
-
return
|
|
107
|
+
return THREE.mergeBufferGeometries(geoms);
|
|
122
108
|
}
|
|
123
109
|
const frustumLength = this.inputs.farPlane - this.inputs.nearPlane;
|
|
124
|
-
const boxGeometry = new BoxGeometry(2, 2, frustumLength);
|
|
110
|
+
const boxGeometry = new THREE.BoxGeometry(2, 2, frustumLength);
|
|
125
111
|
const halfHAngle = this.inputs.horizontalFOV * 0.5 * Math.PI / 180;
|
|
126
112
|
const nearHalfWidth = Math.tan(halfHAngle) * this.inputs.nearPlane;
|
|
127
113
|
const farHalfWidth = Math.tan(halfHAngle) * this.inputs.farPlane;
|
|
@@ -144,41 +130,41 @@ export class SecurityCamera extends SceneComponent {
|
|
|
144
130
|
}
|
|
145
131
|
positions.setZ(i, vertexZ - 0.5 * frustumLength - this.inputs.nearPlane);
|
|
146
132
|
}
|
|
147
|
-
var boxMaterial = new MeshBasicMaterial({
|
|
133
|
+
var boxMaterial = new THREE.MeshBasicMaterial({
|
|
148
134
|
color: this.inputs.color,
|
|
149
135
|
opacity: 0.05,
|
|
150
136
|
transparent: true,
|
|
151
|
-
side: BackSide,
|
|
152
|
-
blending: NormalBlending,
|
|
137
|
+
side: THREE.BackSide,
|
|
138
|
+
blending: THREE.NormalBlending,
|
|
153
139
|
depthWrite: false,
|
|
154
140
|
});
|
|
155
|
-
this.box = new Mesh(boxGeometry, boxMaterial);
|
|
156
|
-
const edgesGeometry = edgesToCylinders(new EdgesGeometry(boxGeometry), 0.015);
|
|
157
|
-
this.edges = new Mesh(edgesGeometry, new MeshBasicMaterial({
|
|
141
|
+
this.box = new THREE.Mesh(boxGeometry, boxMaterial);
|
|
142
|
+
const edgesGeometry = edgesToCylinders(new THREE.EdgesGeometry(boxGeometry), 0.015);
|
|
143
|
+
this.edges = new THREE.Mesh(edgesGeometry, new THREE.MeshBasicMaterial({
|
|
158
144
|
color: 0xffffff,
|
|
159
145
|
opacity: 0.25,
|
|
160
146
|
transparent: true,
|
|
161
|
-
blending: AdditiveBlending,
|
|
147
|
+
blending: THREE.AdditiveBlending,
|
|
162
148
|
depthWrite: false,
|
|
163
149
|
}));
|
|
164
|
-
const edgesGeometry2 = edgesToCylinders(new EdgesGeometry(boxGeometry), 0.06);
|
|
165
|
-
const edges2 = new Mesh(edgesGeometry2, new MeshBasicMaterial({
|
|
150
|
+
const edgesGeometry2 = edgesToCylinders(new THREE.EdgesGeometry(boxGeometry), 0.06);
|
|
151
|
+
const edges2 = new THREE.Mesh(edgesGeometry2, new THREE.MeshBasicMaterial({
|
|
166
152
|
color: this.inputs.color,
|
|
167
153
|
opacity: 0.05,
|
|
168
154
|
transparent: true,
|
|
169
|
-
blending: AdditiveBlending,
|
|
155
|
+
blending: THREE.AdditiveBlending,
|
|
170
156
|
depthWrite: false,
|
|
171
157
|
}));
|
|
172
158
|
this.pivot.add(edges2);
|
|
173
159
|
}
|
|
174
160
|
makeHighlight() {
|
|
161
|
+
const THREE = this.context.three;
|
|
175
162
|
// TODO (scene query): this is very brittle and hardcoded but is the only way to get room geometries until we have a better way to query the scene graph
|
|
176
|
-
|
|
177
|
-
const floorMesh = this.rootScene.getObjectByName('FloorMesh:0');
|
|
163
|
+
const floorMesh = this.outputs.objectRoot.parent.parent.getObjectByName('FloorMesh:0');
|
|
178
164
|
const roomMesh = floorMesh.children[0];
|
|
179
165
|
updateHighlightUniforms(this.projector, this.highlightUniforms);
|
|
180
166
|
this.highlightUniforms.color.value.setHex(this.inputs.color);
|
|
181
|
-
const shader = new ShaderMaterial({
|
|
167
|
+
const shader = new THREE.ShaderMaterial({
|
|
182
168
|
polygonOffset: true,
|
|
183
169
|
polygonOffsetUnits: -0.1,
|
|
184
170
|
transparent: true,
|
|
@@ -187,18 +173,23 @@ export class SecurityCamera extends SceneComponent {
|
|
|
187
173
|
fragmentShader: fragmentShader(),
|
|
188
174
|
});
|
|
189
175
|
// a mesh to represent the "highlight"
|
|
190
|
-
this.highlight = new Mesh(roomMesh.geometry, shader);
|
|
176
|
+
this.highlight = new THREE.Mesh(roomMesh.geometry, shader);
|
|
177
|
+
}
|
|
178
|
+
toggleViewFrustum() {
|
|
179
|
+
this.highlight.visible = !this.highlight.visible;
|
|
180
|
+
this.edges.visible = !this.edges.visible;
|
|
181
|
+
this.pivot.visible = !this.pivot.visible;
|
|
191
182
|
}
|
|
192
183
|
makeAnimation() {
|
|
184
|
+
const THREE = this.context.three;
|
|
193
185
|
// preapply the initial rotation of the root so that panAngle is relative to it.
|
|
194
186
|
// TODO: fix it!
|
|
195
|
-
const rootRotation = this.context.root.quaternion;
|
|
196
|
-
|
|
197
|
-
const
|
|
198
|
-
const
|
|
199
|
-
const frame1 = new Quaternion().setFromAxisAngle(yAxis, this.inputs.panAngle).premultiply(rootRotation);
|
|
187
|
+
const rootRotation = this.context.root.quaternion;
|
|
188
|
+
const yAxis = new THREE.Vector3(0, 1, 0);
|
|
189
|
+
const frame0 = new THREE.Quaternion().setFromAxisAngle(yAxis, 0).premultiply(rootRotation);
|
|
190
|
+
const frame1 = new THREE.Quaternion().setFromAxisAngle(yAxis, this.inputs.panAngle).premultiply(rootRotation);
|
|
200
191
|
// a track that has delays at the start and end for 5% of the animation
|
|
201
|
-
const track = new QuaternionKeyframeTrack('.quaternion', [
|
|
192
|
+
const track = new THREE.QuaternionKeyframeTrack('.quaternion', [
|
|
202
193
|
0, this.inputs.panPeriod * 0.05, this.inputs.panPeriod * 0.95, this.inputs.panPeriod
|
|
203
194
|
], [
|
|
204
195
|
frame0.x, frame0.y, frame0.z, frame0.w,
|
|
@@ -206,13 +197,12 @@ export class SecurityCamera extends SceneComponent {
|
|
|
206
197
|
frame1.x, frame1.y, frame1.z, frame1.w,
|
|
207
198
|
frame1.x, frame1.y, frame1.z, frame1.w,
|
|
208
199
|
]);
|
|
209
|
-
const clip = new AnimationClip('panning', this.inputs.panPeriod, [track]);
|
|
200
|
+
const clip = new THREE.AnimationClip('panning', this.inputs.panPeriod, [track]);
|
|
210
201
|
// There is no api to change the rotation of the root, so we will use a private property to access the Object3D.
|
|
211
202
|
// This will likely need to change in the future.
|
|
212
|
-
this.mixer = new AnimationMixer(this.context.root.obj3D);
|
|
213
|
-
// this.mixer = new AnimationMixer(this.root);
|
|
203
|
+
this.mixer = new THREE.AnimationMixer(this.context.root.obj3D);
|
|
214
204
|
const action = this.mixer.clipAction(clip);
|
|
215
|
-
action.loop = LoopPingPong;
|
|
205
|
+
action.loop = THREE.LoopPingPong;
|
|
216
206
|
action.play();
|
|
217
207
|
}
|
|
218
208
|
}
|
|
@@ -256,4 +246,4 @@ export const securityCameraType = 'mp.securityCamera';
|
|
|
256
246
|
export const makeSecurityCamera = function () {
|
|
257
247
|
return new SecurityCamera();
|
|
258
248
|
};
|
|
259
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"SecurityCamera.js","sourceRoot":"","sources":["../../../../../../projects/ngx-smarterplan-core/src/lib/matterport-extensions/security-camera/SecurityCamera.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Object3D,\n  Mesh,\n  PerspectiveCamera,\n  MeshBasicMaterial,\n  BoxGeometry,\n  ShaderMaterial,\n  LineBasicMaterial,\n  EdgesGeometry,\n  AnimationMixer,\n  Color,\n  Matrix4,\n  Vector3,\n  Euler,\n  CylinderBufferGeometry,\n  BackSide,\n  NormalBlending,\n  AdditiveBlending,\n  Quaternion,\n  QuaternionKeyframeTrack,\n  AnimationClip,\n  LoopPingPong\n} from 'three';\nimport * as BufferGeometryUtils from 'three/examples/jsm/utils/BufferGeometryUtils.js'\nimport {ComponentContext, SceneComponent} from '../scene-component/SceneComponent';\n\ninterface Inputs {\n  nearPlane: number;\n  farPlane: number,\n  horizontalFOV: number,\n  aspect: number,\n  localPosition: { x: number; y: number; z: number; };\n  localRotation: { x: number; y: number; z: number; };\n  color: number;\n  panPeriod: number;\n  panAngle: number;\n}\n\ntype HighlightUniforms = {\n  color: { value: Color },\n  projPosition: { value: Vector3 };\n  lightMatrix: { value: Matrix4 };\n}\n\n\nexport class SecurityCamera extends SceneComponent {\n  private rootScene: Object3D | null = null;\n  private root: Object3D | null = null;\n  private pivot: Object3D | null = null;\n  private box: Mesh | null = null;\n  private edges: Mesh | null = null;\n  private projector: PerspectiveCamera;\n  private highlight: Mesh | null = null;\n  private highlightUniforms: HighlightUniforms = {\n    color: {value: new Color()},\n    projPosition: {value: new Vector3()},\n    lightMatrix: {value: new Matrix4()},\n  }\n  private mixer: AnimationMixer;\n\n  override inputs: Inputs = {\n    nearPlane: 0.1,\n    farPlane: 4,\n    horizontalFOV: 80,\n    aspect: 16 / 9,\n    localPosition: {x: 0, y: 0, z: 0},\n    localRotation: {x: 0, y: 0, z: 0},\n    color: 0xffffff,\n    panPeriod: 20,\n    panAngle: 180,\n  };\n\n  override outputs: any = {};\n  private component: SceneComponent;\n\n\n  setComponent(component: SceneComponent) {\n    component.onTick = this.onTick.bind(this);\n    this.component = component;\n    this.context = component.context;\n  }\n\n  setRootScene(rootScene: Object3D) {\n    this.rootScene = rootScene;\n  }\n\n  override onInit(modelNode?: any) {\n    if (!this.context) {\n      this.context = {\n        root: modelNode\n      } as ComponentContext;\n    }\n\n    this.root = new Object3D();\n    this.pivot = new Object3D();\n    this.root.add(this.pivot);\n    this.outputs.objectRoot = this.root;\n\n    this.root.position.set(this.inputs.localPosition.x, this.inputs.localPosition.y, this.inputs.localPosition.z);\n\n    const euler = new Euler(this.inputs.localRotation.x * Math.PI / 180, this.inputs.localRotation.y * Math.PI / 180, this.inputs.localRotation.z * Math.PI / 180, 'YXZ');\n    this.pivot.quaternion.setFromEuler(euler);\n\n    const aspect = this.inputs.aspect;\n    const DEG2RAD = Math.PI / 180;\n    const RAD2DEG = 1 / DEG2RAD;\n    const verticalFOV = 2 * Math.atan(1 / aspect * Math.tan(0.5 * this.inputs.horizontalFOV * DEG2RAD)) * RAD2DEG;\n    this.projector = new PerspectiveCamera(verticalFOV, aspect, this.inputs.nearPlane, this.inputs.farPlane);\n    // orientation of the projector is handled by the pivot\n    this.pivot.add(this.projector);\n\n    this.makeFrustumVisuals();\n\n    this.pivot.add(this.box);\n    this.pivot.add(this.edges);\n\n    this.makeHighlight();\n\n    // attach highighting to the mesh it represents\n    // TODO (scene query): this is very brittle but is the only way to attach something to a room of the model\n    // this.outputs.objectRoot.parent.parent.add(this.highlight); // legacy matterport code\n    this.rootScene.add(this.highlight);\n\n    // create the animation mixer to rotate the camera\n    this.makeAnimation();\n  }\n\n\n  override onTick(delta: number) {\n    updateHighlightUniforms(this.projector, this.highlightUniforms);\n    if (this.mixer) {\n      this.mixer.update(delta / 1000);\n    }\n  }\n\n  override onDestroy() {\n    if (this.highlight && this.highlight.parent) {\n      this.highlight.parent.remove(this.highlight);\n      (this.highlight.material as ShaderMaterial).dispose();\n    }\n    if (this.edges) {\n      this.edges.geometry.dispose();\n      (this.edges.material as LineBasicMaterial).dispose();\n    }\n    if (this.box) {\n      this.box.geometry.dispose();\n      (this.box.material as MeshBasicMaterial).dispose();\n    }\n  }\n\n  private makeFrustumVisuals() {\n    if (this.box) {\n      this.root.remove(this.box);\n      (this.box.material as MeshBasicMaterial).dispose();\n      (this.box.geometry as BoxGeometry).dispose();\n      this.box = null;\n    }\n    if (this.edges) {\n      this.root.remove(this.edges);\n      (this.edges.material as LineBasicMaterial).dispose();\n      (this.edges.geometry as EdgesGeometry).dispose();\n      this.edges = null;\n    }\n\n    function edgesToCylinders(edgesGeometry: EdgesGeometry, thickness: number) {\n      const {position} = edgesGeometry.attributes;\n      const {array, count} = position;\n      const r = thickness / 2;\n      const geoms = [];\n      for (let i = 0; i < count * 3 - 1; i += 6) {\n        const a = new Vector3(array[i], array[i + 1], array[i + 2]);\n        const b = new Vector3(array[i + 3], array[i + 4], array[i + 5]);\n\n        const vec = new Vector3().subVectors(b, a);\n        const len = vec.length();\n        const geom = new CylinderBufferGeometry(r, r, len, 8);\n        geom.translate(0, len / 2, 0);\n        geom.rotateX(Math.PI / 2);\n        geom.lookAt(vec);\n        geom.translate(a.x, a.y, a.z);\n        geoms.push(geom);\n      }\n      // API to be updated in bundle - JSSDK-1222\n      return BufferGeometryUtils.mergeBufferGeometries(geoms);\n    }\n\n    const frustumLength = this.inputs.farPlane - this.inputs.nearPlane;\n    const boxGeometry: BoxGeometry = new BoxGeometry(2, 2, frustumLength);\n    const halfHAngle = this.inputs.horizontalFOV * 0.5 * Math.PI / 180;\n    const nearHalfWidth = Math.tan(halfHAngle) * this.inputs.nearPlane;\n    const farHalfWidth = Math.tan(halfHAngle) * this.inputs.farPlane;\n    const nearHalfHeight = nearHalfWidth / this.inputs.aspect;\n    const farHalfHeight = farHalfWidth / this.inputs.aspect;\n\n    const positions = boxGeometry.getAttribute('position');\n\n    for (let i = 0; i < positions.count; i++) {\n      const vertexZ = positions.getZ(i);\n      const vertexX = positions.getX(i);\n      const vertexY = positions.getY(i);\n      if (vertexZ > 0) {\n        // back of the camera\n        positions.setX(i, vertexX * nearHalfWidth)\n        positions.setY(i, vertexY * nearHalfHeight);\n      } else {\n        // front of the camera\n        positions.setX(i, vertexX * farHalfWidth)\n        positions.setY(i, vertexY * farHalfHeight);\n      }\n      positions.setZ(i, vertexZ - 0.5 * frustumLength - this.inputs.nearPlane);\n    }\n\n    var boxMaterial: MeshBasicMaterial = new MeshBasicMaterial({\n      color: this.inputs.color,\n      opacity: 0.05,\n      transparent: true,\n      side: BackSide,\n      blending: NormalBlending,\n      depthWrite: false,\n    });\n    this.box = new Mesh(boxGeometry, boxMaterial);\n    const edgesGeometry = edgesToCylinders(new EdgesGeometry(boxGeometry), 0.015);\n    this.edges = new Mesh(edgesGeometry, new MeshBasicMaterial({\n      color: 0xffffff,\n      opacity: 0.25,\n      transparent: true,\n      blending: AdditiveBlending,\n      depthWrite: false,\n    }));\n\n    const edgesGeometry2 = edgesToCylinders(new EdgesGeometry(boxGeometry), 0.06);\n    const edges2 = new Mesh(edgesGeometry2, new MeshBasicMaterial({\n      color: this.inputs.color,\n      opacity: 0.05,\n      transparent: true,\n      blending: AdditiveBlending,\n      depthWrite: false,\n    }));\n    this.pivot.add(edges2);\n  }\n\n  private makeHighlight() {\n\n    // TODO (scene query): this is very brittle and hardcoded but is the only way to get room geometries until we have a better way to query the scene graph\n    // const floorMesh = this.outputs.objectRoot.parent.parent.getObjectByName('FloorMesh:0'); // legacy matterport code..\n    const floorMesh = this.rootScene.getObjectByName('FloorMesh:0');\n    const roomMesh = floorMesh.children[0] as Mesh;\n\n    updateHighlightUniforms(this.projector, this.highlightUniforms);\n    this.highlightUniforms.color.value.setHex(this.inputs.color);\n\n    const shader = new ShaderMaterial({\n      polygonOffset: true,\n      polygonOffsetUnits: -0.1,\n      transparent: true,\n      uniforms: this.highlightUniforms,\n      vertexShader: vertexShader(),\n      fragmentShader: fragmentShader(),\n    });\n\n    // a mesh to represent the \"highlight\"\n    this.highlight = new Mesh(roomMesh.geometry, shader);\n\n\n  }\n\n  private makeAnimation() {\n\n    // preapply the initial rotation of the root so that panAngle is relative to it.\n\n    // TODO: fix it!\n    const rootRotation = this.context.root.quaternion; // legacy matterport code\n    // const rootRotation = this.root.quaternion;\n\n    const yAxis = new Vector3(0, 1, 0);\n    const frame0 = new Quaternion().setFromAxisAngle(yAxis, 0).premultiply(rootRotation);\n    const frame1 = new Quaternion().setFromAxisAngle(yAxis, this.inputs.panAngle).premultiply(rootRotation);\n\n    // a track that has delays at the start and end for 5% of the animation\n    const track = new QuaternionKeyframeTrack('.quaternion', [\n      0, this.inputs.panPeriod * 0.05, this.inputs.panPeriod * 0.95, this.inputs.panPeriod\n    ], [\n      frame0.x, frame0.y, frame0.z, frame0.w,\n      frame0.x, frame0.y, frame0.z, frame0.w,\n      frame1.x, frame1.y, frame1.z, frame1.w,\n      frame1.x, frame1.y, frame1.z, frame1.w,\n    ]);\n\n    const clip = new AnimationClip('panning', this.inputs.panPeriod, [track]);\n\n    // There is no api to change the rotation of the root, so we will use a private property to access the Object3D.\n    // This will likely need to change in the future.\n    this.mixer = new AnimationMixer((this.context.root as any).obj3D); // legacode code matterport\n    // this.mixer = new AnimationMixer(this.root);\n    const action = this.mixer.clipAction(clip);\n    action.loop = LoopPingPong;\n    action.play();\n  }\n\n}\n\nfunction vertexShader(): string {\n  return `\n    uniform mat4 lightMatrix;\n\n    varying vec4 vLightCoords;\n\n    void main() {\n      // create a UV region [0,1] x [0,1] to represent what the lightMatrix \"sees\"\n      vLightCoords = lightMatrix *  modelMatrix * vec4(position, 1.0);\n      gl_Position = projectionMatrix * modelViewMatrix * vec4(position, 1.0);\n    }\n  `;\n}\n\nfunction fragmentShader() {\n  return `\n    uniform vec3 color;\n\n    varying vec4 vLightCoords;\n\n    void main() {\n      vec2 lightUV = vLightCoords.xy / vLightCoords.w;\n      float inView = float(max(lightUV.x, lightUV.y) <= 1.0 && min(lightUV.x, lightUV.y) >= 0.0 && vLightCoords.z > 0.0);\n\n      gl_FragColor = vec4(color, min(inView, 0.2));\n    }\n  `;\n}\n\nfunction updateHighlightUniforms(projector: PerspectiveCamera, uniforms: HighlightUniforms) {\n  projector.getWorldPosition(uniforms.projPosition.value);\n\n  // similar to a shadow matrix, but we're using it as a \"rectangular\" spotlight\n  // [lightBias] * [projection] * [viewMatrix]\n  uniforms.lightMatrix.value.makeScale(0.5, 0.5, 0.5);\n  uniforms.lightMatrix.value.setPosition(new Vector3(0.5, 0.5, 0.5))\n  uniforms.lightMatrix.value.multiplyMatrices(uniforms.lightMatrix.value, projector.projectionMatrix);\n  uniforms.lightMatrix.value.multiplyMatrices(uniforms.lightMatrix.value, projector.matrixWorldInverse);\n}\n\nexport const securityCameraType = 'mp.securityCamera';\nexport const makeSecurityCamera = function () {\n  return new SecurityCamera();\n}\n"]}
|
|
249
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"SecurityCamera.js","sourceRoot":"","sources":["../../../../../../projects/ngx-smarterplan-core/src/lib/matterport-extensions/security-camera/SecurityCamera.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Object3D, Mesh, PerspectiveCamera,\n  MeshBasicMaterial, BoxGeometry, ShaderMaterial, LineBasicMaterial, EdgesGeometry,\n  AnimationMixer, Color, Matrix4, Vector3\n} from 'three';\nimport {SceneComponent, ISceneNode} from '../scene-component/SceneComponent';\n\ndeclare global {\n  interface Window {\n    THREE: any;\n  }\n}\n\ninterface Inputs {\n  nearPlane: number;\n  farPlane: number,\n  horizontalFOV: number,\n  aspect: number,\n  localPosition: { x: number; y: number; z: number; };\n  localRotation: { x: number; y: number; z: number; };\n  color: number;\n  panPeriod: number;\n  panAngle: number;\n}\n\ntype HighlightUniforms = {\n  color: { value: Color },\n  projPosition: { value: Vector3 };\n  lightMatrix: { value: Matrix4 };\n}\n\n\nexport class SecurityCamera extends SceneComponent {\n  private root: Object3D | null = null;\n  private pivot: Object3D | null = null;\n  private box: Mesh | null = null;\n  private edges: Mesh | null = null;\n  private projector: PerspectiveCamera;\n  private highlight: Mesh | null = null;\n  private highlightUniforms: HighlightUniforms = {\n    color: {value: new Color()},\n    projPosition: {value: new Vector3()},\n    lightMatrix: {value: new Matrix4()},\n  }\n  private mixer: AnimationMixer;\n\n  override inputs: Inputs = {\n    nearPlane: 0.1,\n    farPlane: 4,\n    horizontalFOV: 80,\n    aspect: 16 / 9,\n    localPosition: {x: 0, y: 0, z: 0},\n    localRotation: {x: 0, y: 0, z: 0},\n    color: 0xffffff,\n    panPeriod: 20,\n    panAngle: 180,\n  };\n\n  override outputs: any = {};\n\n\n  override onInit() {\n    const THREE = this.context.three;\n\n    this.root = new THREE.Object3D();\n    this.pivot = new THREE.Object3D();\n    this.root.add(this.pivot);\n    this.outputs.objectRoot = this.root;\n    this.root.position.set(this.inputs.localPosition.x, this.inputs.localPosition.y, this.inputs.localPosition.z);\n\n    const euler = new THREE.Euler(this.inputs.localRotation.x * Math.PI / 180, this.inputs.localRotation.y * Math.PI / 180, this.inputs.localRotation.z * Math.PI / 180, 'YXZ');\n    this.pivot.quaternion.setFromEuler(euler);\n\n    const aspect = this.inputs.aspect;\n    const DEG2RAD = Math.PI / 180;\n    const RAD2DEG = 1 / DEG2RAD;\n    const verticalFOV = 2 * Math.atan(1 / aspect * Math.tan(0.5 * this.inputs.horizontalFOV * DEG2RAD)) * RAD2DEG;\n    this.projector = new THREE.PerspectiveCamera(verticalFOV, aspect, this.inputs.nearPlane, this.inputs.farPlane);\n    // orientation of the projector is handled by the pivot\n    this.pivot.add(this.projector);\n\n    this.makeFrustumVisuals();\n\n    this.pivot.add(this.box);\n    this.pivot.add(this.edges);\n\n    this.makeHighlight();\n\n    // attach highighting to the mesh it represents\n    // TODO (scene query): this is very brittle but is the only way to attach something to a room of the model\n    this.outputs.objectRoot.parent.parent.add(this.highlight);\n\n    // create the animation mixer to rotate the camera\n    this.makeAnimation();\n  }\n\n  override onTick(delta: number) {\n    updateHighlightUniforms(this.projector, this.highlightUniforms);\n    if (this.mixer) {\n      this.mixer.update(delta / 1000);\n    }\n  }\n\n  override onDestroy() {\n    if (this.highlight && this.highlight.parent) {\n      this.highlight.parent.remove(this.highlight);\n      (this.highlight.material as ShaderMaterial).dispose();\n    }\n    if (this.edges) {\n      this.edges.geometry.dispose();\n      (this.edges.material as LineBasicMaterial).dispose();\n    }\n    if (this.box) {\n      this.box.geometry.dispose();\n      (this.box.material as MeshBasicMaterial).dispose();\n    }\n\n  }\n\n  private makeFrustumVisuals() {\n    const THREE = this.context.three;\n    if (this.box) {\n      this.root.remove(this.box);\n      (this.box.material as MeshBasicMaterial).dispose();\n      (this.box.geometry as BoxGeometry).dispose();\n      this.box = null;\n    }\n    if (this.edges) {\n      this.root.remove(this.edges);\n      (this.edges.material as LineBasicMaterial).dispose();\n      (this.edges.geometry as EdgesGeometry).dispose();\n      this.edges = null;\n    }\n\n    function edgesToCylinders(edgesGeometry: EdgesGeometry, thickness: number) {\n      const {position} = edgesGeometry.attributes;\n      const {array, count} = position;\n      const r = thickness / 2;\n      const geoms = [];\n      for (let i = 0; i < count * 3 - 1; i += 6) {\n        const a = new THREE.Vector3(array[i], array[i + 1], array[i + 2]);\n        const b = new THREE.Vector3(array[i + 3], array[i + 4], array[i + 5]);\n\n        const vec = new THREE.Vector3().subVectors(b, a);\n        const len = vec.length();\n        const geom = new THREE.CylinderBufferGeometry(r, r, len, 8);\n        geom.translate(0, len / 2, 0);\n        geom.rotateX(Math.PI / 2);\n        geom.lookAt(vec);\n        geom.translate(a.x, a.y, a.z);\n        geoms.push(geom);\n      }\n      // API to be updated in bundle - JSSDK-1222\n      return (THREE as any).mergeBufferGeometries(geoms);\n    }\n\n    const frustumLength = this.inputs.farPlane - this.inputs.nearPlane;\n    const boxGeometry: BoxGeometry = new THREE.BoxGeometry(2, 2, frustumLength);\n    const halfHAngle = this.inputs.horizontalFOV * 0.5 * Math.PI / 180;\n    const nearHalfWidth = Math.tan(halfHAngle) * this.inputs.nearPlane;\n    const farHalfWidth = Math.tan(halfHAngle) * this.inputs.farPlane;\n    const nearHalfHeight = nearHalfWidth / this.inputs.aspect;\n    const farHalfHeight = farHalfWidth / this.inputs.aspect;\n\n    const positions = boxGeometry.getAttribute('position');\n\n    for (let i = 0; i < positions.count; i++) {\n      const vertexZ = positions.getZ(i);\n      const vertexX = positions.getX(i);\n      const vertexY = positions.getY(i);\n      if (vertexZ > 0) {\n        // back of the camera\n        positions.setX(i, vertexX * nearHalfWidth)\n        positions.setY(i, vertexY * nearHalfHeight);\n      } else {\n        // front of the camera\n        positions.setX(i, vertexX * farHalfWidth)\n        positions.setY(i, vertexY * farHalfHeight);\n      }\n      positions.setZ(i, vertexZ - 0.5 * frustumLength - this.inputs.nearPlane);\n    }\n\n    var boxMaterial: MeshBasicMaterial = new THREE.MeshBasicMaterial({\n      color: this.inputs.color,\n      opacity: 0.05,\n      transparent: true,\n      side: THREE.BackSide,\n      blending: THREE.NormalBlending,\n      depthWrite: false,\n    });\n    this.box = new THREE.Mesh(boxGeometry, boxMaterial);\n    const edgesGeometry = edgesToCylinders(new THREE.EdgesGeometry(boxGeometry), 0.015);\n    this.edges = new THREE.Mesh(edgesGeometry, new THREE.MeshBasicMaterial({\n      color: 0xffffff,\n      opacity: 0.25,\n      transparent: true,\n      blending: THREE.AdditiveBlending,\n      depthWrite: false,\n    }));\n\n    const edgesGeometry2 = edgesToCylinders(new THREE.EdgesGeometry(boxGeometry), 0.06);\n    const edges2 = new THREE.Mesh(edgesGeometry2, new THREE.MeshBasicMaterial({\n      color: this.inputs.color,\n      opacity: 0.05,\n      transparent: true,\n      blending: THREE.AdditiveBlending,\n      depthWrite: false,\n    }));\n    this.pivot.add(edges2);\n  }\n\n  private makeHighlight() {\n    const THREE = this.context.three;\n\n    // TODO (scene query): this is very brittle and hardcoded but is the only way to get room geometries until we have a better way to query the scene graph\n    const floorMesh = this.outputs.objectRoot.parent.parent.getObjectByName('FloorMesh:0');\n    const roomMesh = floorMesh.children[0] as Mesh;\n\n    updateHighlightUniforms(this.projector, this.highlightUniforms);\n    this.highlightUniforms.color.value.setHex(this.inputs.color);\n\n    const shader = new THREE.ShaderMaterial({\n      polygonOffset: true,\n      polygonOffsetUnits: -0.1,\n      transparent: true,\n      uniforms: this.highlightUniforms,\n      vertexShader: vertexShader(),\n      fragmentShader: fragmentShader(),\n    });\n\n    // a mesh to represent the \"highlight\"\n    this.highlight = new THREE.Mesh(roomMesh.geometry, shader);\n  }\n\n  toggleViewFrustum(): void {\n    this.highlight.visible = !this.highlight.visible;\n    this.edges.visible = !this.edges.visible;\n    this.pivot.visible = !this.pivot.visible\n  }\n\n  private makeAnimation() {\n    const THREE = this.context.three;\n\n    // preapply the initial rotation of the root so that panAngle is relative to it.\n\n    // TODO: fix it!\n    const rootRotation = this.context.root.quaternion;\n\n    const yAxis = new THREE.Vector3(0, 1, 0);\n    const frame0 = new THREE.Quaternion().setFromAxisAngle(yAxis, 0).premultiply(rootRotation);\n    const frame1 = new THREE.Quaternion().setFromAxisAngle(yAxis, this.inputs.panAngle).premultiply(rootRotation);\n\n    // a track that has delays at the start and end for 5% of the animation\n    const track = new THREE.QuaternionKeyframeTrack('.quaternion', [\n      0, this.inputs.panPeriod * 0.05, this.inputs.panPeriod * 0.95, this.inputs.panPeriod\n    ], [\n      frame0.x, frame0.y, frame0.z, frame0.w,\n      frame0.x, frame0.y, frame0.z, frame0.w,\n      frame1.x, frame1.y, frame1.z, frame1.w,\n      frame1.x, frame1.y, frame1.z, frame1.w,\n    ]);\n\n    const clip = new THREE.AnimationClip('panning', this.inputs.panPeriod, [track]);\n\n    // There is no api to change the rotation of the root, so we will use a private property to access the Object3D.\n    // This will likely need to change in the future.\n    this.mixer = new THREE.AnimationMixer((this.context.root as any).obj3D);\n    const action = this.mixer.clipAction(clip);\n    action.loop = THREE.LoopPingPong;\n    action.play();\n  }\n\n}\n\nfunction vertexShader(): string {\n  return `\n    uniform mat4 lightMatrix;\n\n    varying vec4 vLightCoords;\n\n    void main() {\n      // create a UV region [0,1] x [0,1] to represent what the lightMatrix \"sees\"\n      vLightCoords = lightMatrix *  modelMatrix * vec4(position, 1.0);\n      gl_Position = projectionMatrix * modelViewMatrix * vec4(position, 1.0);\n    }\n  `;\n}\n\nfunction fragmentShader() {\n  return `\n    uniform vec3 color;\n\n    varying vec4 vLightCoords;\n\n    void main() {\n      vec2 lightUV = vLightCoords.xy / vLightCoords.w;\n      float inView = float(max(lightUV.x, lightUV.y) <= 1.0 && min(lightUV.x, lightUV.y) >= 0.0 && vLightCoords.z > 0.0);\n\n      gl_FragColor = vec4(color, min(inView, 0.2));\n    }\n  `;\n}\n\nfunction updateHighlightUniforms(projector: PerspectiveCamera, uniforms: HighlightUniforms) {\n  projector.getWorldPosition(uniforms.projPosition.value);\n\n  // similar to a shadow matrix, but we're using it as a \"rectangular\" spotlight\n  // [lightBias] * [projection] * [viewMatrix]\n  uniforms.lightMatrix.value.makeScale(0.5, 0.5, 0.5);\n  uniforms.lightMatrix.value.setPosition(new Vector3(0.5, 0.5, 0.5))\n  uniforms.lightMatrix.value.multiplyMatrices(uniforms.lightMatrix.value, projector.projectionMatrix);\n  uniforms.lightMatrix.value.multiplyMatrices(uniforms.lightMatrix.value, projector.matrixWorldInverse);\n}\n\nexport const securityCameraType = 'mp.securityCamera';\nexport const makeSecurityCamera = function () {\n  return new SecurityCamera();\n}\n"]}
|
|
@@ -0,0 +1,98 @@
|
|
|
1
|
+
import { SceneComponent } from '../scene-component/SceneComponent';
|
|
2
|
+
import { Texture, MeshBasicMaterial, LinearFilter, } from 'three';
|
|
3
|
+
const HoverEvent = 'hover';
|
|
4
|
+
const UnhoverEvent = 'unhover';
|
|
5
|
+
export class TvPlayer extends SceneComponent {
|
|
6
|
+
constructor() {
|
|
7
|
+
super(...arguments);
|
|
8
|
+
this.rootScene = null;
|
|
9
|
+
this.mesh = null;
|
|
10
|
+
this.inputs = {
|
|
11
|
+
loadingState: 'Idle',
|
|
12
|
+
texture: null,
|
|
13
|
+
updateInterval: 1000,
|
|
14
|
+
};
|
|
15
|
+
this.events = {
|
|
16
|
+
[HoverEvent]: true,
|
|
17
|
+
[UnhoverEvent]: true,
|
|
18
|
+
};
|
|
19
|
+
}
|
|
20
|
+
onInit(modelNode) {
|
|
21
|
+
if (!this.context) {
|
|
22
|
+
this.context = {
|
|
23
|
+
root: modelNode
|
|
24
|
+
};
|
|
25
|
+
}
|
|
26
|
+
const root = this.context.root;
|
|
27
|
+
const THREE = this.context.three;
|
|
28
|
+
}
|
|
29
|
+
onDestroy() {
|
|
30
|
+
super.onDestroy();
|
|
31
|
+
clearInterval(this.intervalVideoTexture);
|
|
32
|
+
}
|
|
33
|
+
onInputsUpdated() {
|
|
34
|
+
if (this.inputs.loadingState === 'Loaded') {
|
|
35
|
+
const lines = [];
|
|
36
|
+
//@ts-ignore
|
|
37
|
+
this.context.root.obj3D.traverse((obj) => {
|
|
38
|
+
// we dont want line segments, track them and remove them.
|
|
39
|
+
if (obj.type === 'LineSegments') {
|
|
40
|
+
lines.push(obj);
|
|
41
|
+
}
|
|
42
|
+
else if (obj.type === 'Mesh') {
|
|
43
|
+
this.mesh = obj;
|
|
44
|
+
let video, videoImage, videoImageContext, videoTexture;
|
|
45
|
+
// create the video element
|
|
46
|
+
video = document.createElement('video');
|
|
47
|
+
// video.src = "./../assets/video/video_samsic.mp4";
|
|
48
|
+
video.src = "./../assets/video/121222-04.mp4";
|
|
49
|
+
video.load(); // must call after setting/changing source
|
|
50
|
+
video.loop = true;
|
|
51
|
+
video.play();
|
|
52
|
+
videoImage = document.createElement('canvas');
|
|
53
|
+
// videoImage.width = 640;
|
|
54
|
+
videoImage.width = 1280;
|
|
55
|
+
// videoImage.height = 360;
|
|
56
|
+
videoImage.height = 720;
|
|
57
|
+
videoImageContext = videoImage.getContext('2d');
|
|
58
|
+
// background color if no video present
|
|
59
|
+
videoImageContext.fillStyle = '#000000';
|
|
60
|
+
videoImageContext.fillRect(0, 0, videoImage.width, videoImage.height);
|
|
61
|
+
videoTexture = new Texture(videoImage);
|
|
62
|
+
videoTexture.minFilter = LinearFilter;
|
|
63
|
+
videoTexture.magFilter = LinearFilter;
|
|
64
|
+
const movieMaterial = new MeshBasicMaterial({ map: videoTexture });
|
|
65
|
+
this.mesh.material = movieMaterial;
|
|
66
|
+
const render = () => {
|
|
67
|
+
if (video.readyState === video.HAVE_ENOUGH_DATA) {
|
|
68
|
+
videoImageContext.drawImage(video, 0, 0);
|
|
69
|
+
if (videoTexture)
|
|
70
|
+
videoTexture.needsUpdate = true;
|
|
71
|
+
}
|
|
72
|
+
const camera = (this.context.scene.children.find(u => u.constructor.name === 'CameraRig')).camera;
|
|
73
|
+
this.context.renderer.render(this.context.scene, camera);
|
|
74
|
+
};
|
|
75
|
+
this.intervalVideoTexture = setInterval(() => {
|
|
76
|
+
render();
|
|
77
|
+
}, 180);
|
|
78
|
+
}
|
|
79
|
+
});
|
|
80
|
+
// remove the line segments.
|
|
81
|
+
lines.forEach((line) => {
|
|
82
|
+
line.parent.remove(line);
|
|
83
|
+
});
|
|
84
|
+
}
|
|
85
|
+
}
|
|
86
|
+
setComponent(component) {
|
|
87
|
+
this.component = component;
|
|
88
|
+
this.context = component.context;
|
|
89
|
+
}
|
|
90
|
+
setRootScene(rootScene) {
|
|
91
|
+
this.rootScene = rootScene;
|
|
92
|
+
}
|
|
93
|
+
}
|
|
94
|
+
export const TvPlayerType = 'mp.TvPlayer';
|
|
95
|
+
export const makeTvPlayer = function () {
|
|
96
|
+
return new TvPlayer();
|
|
97
|
+
};
|
|
98
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"TvPlayer.js","sourceRoot":"","sources":["../../../../../../projects/ngx-smarterplan-core/src/lib/matterport-extensions/tv-player/TvPlayer.ts"],"names":[],"mappings":"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","sourcesContent":["import {SceneComponent, ComponentContext} from '../scene-component/SceneComponent';\nimport {\n  Object3D,\n  Mesh,\n  Texture,\n  MeshBasicMaterial,\n  LineSegments,\n  LinearFilter,\n} from 'three';\n\nconst HoverEvent = 'hover';\nconst UnhoverEvent = 'unhover';\n\ntype Inputs = {\n  loadingState: string;\n  texture: Texture | null;\n  updateInterval: number;\n}\n\n\nexport class TvPlayer extends SceneComponent {\n  private rootScene: Object3D | null = null;\n  private mesh: Mesh | null = null;\n  private component: SceneComponent;\n\n\n  override inputs: Inputs = {\n    loadingState: 'Idle',\n    texture: null,\n    updateInterval: 1000,\n  }\n\n\n  override events = {\n    [HoverEvent]: true,\n    [UnhoverEvent]: true,\n  };\n  private intervalVideoTexture: NodeJS.Timeout;\n\n  override onInit(modelNode?: any) {\n    if (!this.context) {\n      this.context = {\n        root: modelNode\n      } as ComponentContext;\n    }\n    const root = this.context.root;\n    const THREE = this.context.three;\n  }\n\n  override onDestroy() {\n    super.onDestroy();\n    clearInterval(this.intervalVideoTexture);\n  }\n\n  override onInputsUpdated() {\n    if (this.inputs.loadingState === 'Loaded') {\n      const lines: LineSegments[] = [];\n\n      //@ts-ignore\n      this.context.root.obj3D.traverse((obj: Object3D) => {\n        // we dont want line segments, track them and remove them.\n        if (obj.type === 'LineSegments') {\n          lines.push(obj as LineSegments);\n        } else if (obj.type === 'Mesh') {\n          this.mesh = obj as Mesh;\n\n          let video, videoImage, videoImageContext, videoTexture;\n          // create the video element\n          video = document.createElement('video');\n          // video.src = \"./../assets/video/video_samsic.mp4\";\n          video.src = \"./../assets/video/121222-04.mp4\";\n          video.load(); // must call after setting/changing source\n          video.loop = true;\n          video.play();\n\n          videoImage = document.createElement('canvas');\n          // videoImage.width = 640;\n          videoImage.width = 1280;\n          // videoImage.height = 360;\n          videoImage.height = 720;\n\n          videoImageContext = videoImage.getContext('2d');\n          // background color if no video present\n          videoImageContext.fillStyle = '#000000';\n          videoImageContext.fillRect(0, 0, videoImage.width, videoImage.height);\n\n          videoTexture = new Texture(videoImage)\n          videoTexture.minFilter = LinearFilter;\n          videoTexture.magFilter = LinearFilter;\n          const movieMaterial = new MeshBasicMaterial({map: videoTexture})\n          this.mesh.material = movieMaterial;\n\n          const render = () => {\n            if (video.readyState === video.HAVE_ENOUGH_DATA) {\n              videoImageContext.drawImage(video, 0, 0);\n              if (videoTexture)\n                videoTexture.needsUpdate = true;\n            }\n            const camera = (this.context.scene.children.find(u => u.constructor.name === 'CameraRig')).camera;\n            this.context.renderer.render(this.context.scene, camera);\n          }\n          this.intervalVideoTexture = setInterval(() => {\n            render();\n          }, 180);\n\n        }\n      });\n\n      // remove the line segments.\n      lines.forEach((line: LineSegments) => {\n        line.parent.remove(line);\n      });\n    }\n  }\n\n  setComponent(component: SceneComponent) {\n    this.component = component;\n    this.context = component.context;\n  }\n\n  setRootScene(rootScene: Object3D) {\n    this.rootScene = rootScene;\n  }\n}\n\nexport const TvPlayerType = 'mp.TvPlayer';\nexport const makeTvPlayer = function () {\n  return new TvPlayer();\n}\n"]}
|