@smarterplan/ngx-smarterplan-core 1.2.45 → 1.2.47
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/README.md +24 -24
- package/esm2020/lib/components/csv-export/csv-export.component.mjs +59 -59
- package/esm2020/lib/components/loader/loader.component.mjs +23 -23
- package/esm2020/lib/components/menu-bar/avatar/avatar.component.mjs +80 -80
- package/esm2020/lib/components/menu-bar/menu-bar.component.mjs +99 -99
- package/esm2020/lib/components/menu-bar/navigation-bar/navigation-bar.component.mjs +384 -384
- package/esm2020/lib/components/menu-bar/range-date-picker/range-date-picker.component.mjs +147 -147
- package/esm2020/lib/components/modal-switch-visit/modal-switch-visit.component.mjs +40 -40
- package/esm2020/lib/components/search-bar/search-bar.component.mjs +63 -63
- package/esm2020/lib/components/support-modal/support-modal.component.mjs +66 -66
- package/esm2020/lib/config.mjs +4 -4
- package/esm2020/lib/helpers.service.mjs +470 -470
- package/esm2020/lib/matterport-extensions/hsl-loader/HlsLoader.mjs +69 -69
- package/esm2020/lib/matterport-extensions/nest-thermostat/CanvasImage.mjs +51 -51
- package/esm2020/lib/matterport-extensions/nest-thermostat/CanvasRenderer.mjs +61 -61
- package/esm2020/lib/matterport-extensions/nest-thermostat/NestThermostat.mjs +158 -158
- package/esm2020/lib/matterport-extensions/nest-thermostat/PlaneRenderer.mjs +85 -85
- package/esm2020/lib/matterport-extensions/scene-component/SceneComponent.mjs +128 -128
- package/esm2020/lib/matterport-extensions/security-camera/SecurityCamera.mjs +249 -249
- package/esm2020/lib/matterport-extensions/tv-player/TvPlayer.mjs +98 -98
- package/esm2020/lib/matterport-extensions/video-renderer/VideoRenderer.mjs +64 -64
- package/esm2020/lib/matterport-extensions/view-frustum-mesh/ViewFrustumMesh.mjs +221 -221
- package/esm2020/lib/mattertagData.mjs +165 -165
- package/esm2020/lib/ngx-smarterplan-core.module.mjs +122 -122
- package/esm2020/lib/ngx-smarterplan-core.service.mjs +14 -14
- package/esm2020/lib/pipes/duration-to-string.pipe.mjs +66 -66
- package/esm2020/lib/pipes/format-date-number-to-digits.pipe.mjs +30 -30
- package/esm2020/lib/pipes/hashtag-from-id.pipe.mjs +26 -26
- package/esm2020/lib/pipes/safe-url.pipe.mjs +20 -20
- package/esm2020/lib/pipes/time-date-to-local-string.pipe.mjs +104 -104
- package/esm2020/lib/pipes/username-from-id.pipe.mjs +29 -29
- package/esm2020/lib/services/amplify-cache.service.mjs +72 -72
- package/esm2020/lib/services/base-tab.service.mjs +24 -24
- package/esm2020/lib/services/baseVisibility.service.mjs +18 -18
- package/esm2020/lib/services/content.service.mjs +135 -135
- package/esm2020/lib/services/filter.service.mjs +599 -599
- package/esm2020/lib/services/intervention.service.mjs +236 -236
- package/esm2020/lib/services/locale.service.mjs +45 -45
- package/esm2020/lib/services/matterport-import.service.mjs +340 -340
- package/esm2020/lib/services/matterport.service.mjs +1587 -1587
- package/esm2020/lib/services/models/affectation.service.mjs +60 -60
- package/esm2020/lib/services/models/base-object.service.mjs +70 -70
- package/esm2020/lib/services/models/capture.service.mjs +34 -34
- package/esm2020/lib/services/models/comment.service.mjs +98 -98
- package/esm2020/lib/services/models/domain.service.mjs +78 -78
- package/esm2020/lib/services/models/equipment.service.mjs +683 -683
- package/esm2020/lib/services/models/event.service.mjs +128 -128
- package/esm2020/lib/services/models/feature.service.mjs +380 -380
- package/esm2020/lib/services/models/hashtag.service.mjs +38 -38
- package/esm2020/lib/services/models/layer.service.mjs +33 -33
- package/esm2020/lib/services/models/measurement.service.mjs +199 -199
- package/esm2020/lib/services/models/mission.service.mjs +206 -206
- package/esm2020/lib/services/models/navigation.service.mjs +92 -92
- package/esm2020/lib/services/models/node.service.mjs +31 -31
- package/esm2020/lib/services/models/object3D.service.mjs +364 -364
- package/esm2020/lib/services/models/operation.service.mjs +59 -59
- package/esm2020/lib/services/models/organisation.service.mjs +73 -73
- package/esm2020/lib/services/models/plan.service.mjs +799 -799
- package/esm2020/lib/services/models/poi.service.mjs +103 -103
- package/esm2020/lib/services/models/profile.service.mjs +58 -58
- package/esm2020/lib/services/models/property.service.mjs +44 -44
- package/esm2020/lib/services/models/space.service.mjs +204 -204
- package/esm2020/lib/services/models/template.service.mjs +41 -41
- package/esm2020/lib/services/models/ticket.service.mjs +526 -526
- package/esm2020/lib/services/models/visit.service.mjs +130 -130
- package/esm2020/lib/services/models/zone.service.mjs +225 -225
- package/esm2020/lib/services/navigator.service.mjs +212 -212
- package/esm2020/lib/services/s3.service.mjs +137 -137
- package/esm2020/lib/services/search.service.mjs +124 -124
- package/esm2020/lib/services/support.service.mjs +42 -42
- package/esm2020/lib/services/tag.service.mjs +111 -111
- package/esm2020/lib/services/user.service.mjs +501 -501
- package/esm2020/lib/services/validators.service.mjs +50 -50
- package/esm2020/lib/services/viewer.service.mjs +389 -389
- package/esm2020/lib/services/zone-drawer.service.mjs +76 -76
- package/esm2020/lib/services/zoneChange.service.mjs +30 -30
- package/esm2020/lib/types.service.mjs +311 -311
- package/esm2020/lib/validators/email.directive.mjs +7 -7
- package/esm2020/lib/validators/no-empty.directive.mjs +12 -12
- package/esm2020/lib/validators/number.directive.mjs +12 -12
- package/esm2020/lib/validators/text.directive.mjs +12 -12
- package/esm2020/public-api.mjs +72 -72
- package/esm2020/smarterplan-ngx-smarterplan-core.mjs +4 -4
- package/fesm2015/smarterplan-ngx-smarterplan-core.mjs +13014 -13014
- package/fesm2015/smarterplan-ngx-smarterplan-core.mjs.map +1 -1
- package/fesm2020/smarterplan-ngx-smarterplan-core.mjs +12263 -12263
- package/fesm2020/smarterplan-ngx-smarterplan-core.mjs.map +1 -1
- package/lib/components/csv-export/csv-export.component.d.ts +18 -18
- package/lib/components/loader/loader.component.d.ts +10 -10
- package/lib/components/menu-bar/avatar/avatar.component.d.ts +21 -21
- package/lib/components/menu-bar/menu-bar.component.d.ts +38 -38
- package/lib/components/menu-bar/navigation-bar/navigation-bar.component.d.ts +73 -73
- package/lib/components/menu-bar/range-date-picker/range-date-picker.component.d.ts +35 -35
- package/lib/components/modal-switch-visit/modal-switch-visit.component.d.ts +22 -22
- package/lib/components/search-bar/search-bar.component.d.ts +16 -16
- package/lib/components/support-modal/support-modal.component.d.ts +26 -26
- package/lib/config.d.ts +22 -22
- package/lib/helpers.service.d.ts +79 -79
- package/lib/matterport-extensions/hsl-loader/HlsLoader.d.ts +26 -26
- package/lib/matterport-extensions/nest-thermostat/CanvasImage.d.ts +31 -31
- package/lib/matterport-extensions/nest-thermostat/CanvasRenderer.d.ts +37 -37
- package/lib/matterport-extensions/nest-thermostat/NestThermostat.d.ts +42 -42
- package/lib/matterport-extensions/nest-thermostat/PlaneRenderer.d.ts +46 -46
- package/lib/matterport-extensions/scene-component/SceneComponent.d.ts +388 -388
- package/lib/matterport-extensions/security-camera/SecurityCamera.d.ts +47 -47
- package/lib/matterport-extensions/tv-player/TvPlayer.d.ts +26 -26
- package/lib/matterport-extensions/video-renderer/VideoRenderer.d.ts +26 -26
- package/lib/matterport-extensions/view-frustum-mesh/ViewFrustumMesh.d.ts +43 -43
- package/lib/mattertagData.d.ts +70 -70
- package/lib/ngx-smarterplan-core.module.d.ts +29 -29
- package/lib/ngx-smarterplan-core.service.d.ts +6 -6
- package/lib/pipes/duration-to-string.pipe.d.ts +12 -12
- package/lib/pipes/format-date-number-to-digits.pipe.d.ts +10 -10
- package/lib/pipes/hashtag-from-id.pipe.d.ts +10 -10
- package/lib/pipes/safe-url.pipe.d.ts +10 -10
- package/lib/pipes/time-date-to-local-string.pipe.d.ts +16 -16
- package/lib/pipes/username-from-id.pipe.d.ts +11 -11
- package/lib/services/amplify-cache.service.d.ts +37 -37
- package/lib/services/base-tab.service.d.ts +10 -10
- package/lib/services/baseVisibility.service.d.ts +9 -9
- package/lib/services/content.service.d.ts +28 -28
- package/lib/services/filter.service.d.ts +60 -60
- package/lib/services/intervention.service.d.ts +25 -25
- package/lib/services/locale.service.d.ts +23 -23
- package/lib/services/matterport-import.service.d.ts +53 -53
- package/lib/services/matterport.service.d.ts +336 -336
- package/lib/services/models/affectation.service.d.ts +14 -14
- package/lib/services/models/base-object.service.d.ts +20 -20
- package/lib/services/models/capture.service.d.ts +13 -13
- package/lib/services/models/comment.service.d.ts +26 -26
- package/lib/services/models/domain.service.d.ts +19 -19
- package/lib/services/models/equipment.service.d.ts +93 -93
- package/lib/services/models/event.service.d.ts +43 -43
- package/lib/services/models/feature.service.d.ts +75 -75
- package/lib/services/models/hashtag.service.d.ts +13 -13
- package/lib/services/models/layer.service.d.ts +11 -11
- package/lib/services/models/measurement.service.d.ts +51 -51
- package/lib/services/models/mission.service.d.ts +39 -39
- package/lib/services/models/navigation.service.d.ts +29 -29
- package/lib/services/models/node.service.d.ts +12 -12
- package/lib/services/models/object3D.service.d.ts +57 -57
- package/lib/services/models/operation.service.d.ts +15 -15
- package/lib/services/models/organisation.service.d.ts +19 -19
- package/lib/services/models/plan.service.d.ts +133 -133
- package/lib/services/models/poi.service.d.ts +25 -25
- package/lib/services/models/profile.service.d.ts +16 -16
- package/lib/services/models/property.service.d.ts +13 -13
- package/lib/services/models/space.service.d.ts +46 -46
- package/lib/services/models/template.service.d.ts +15 -15
- package/lib/services/models/ticket.service.d.ts +93 -93
- package/lib/services/models/visit.service.d.ts +24 -24
- package/lib/services/models/zone.service.d.ts +50 -50
- package/lib/services/navigator.service.d.ts +61 -61
- package/lib/services/s3.service.d.ts +14 -14
- package/lib/services/search.service.d.ts +20 -20
- package/lib/services/support.service.d.ts +17 -17
- package/lib/services/tag.service.d.ts +29 -29
- package/lib/services/user.service.d.ts +118 -118
- package/lib/services/validators.service.d.ts +18 -18
- package/lib/services/viewer.service.d.ts +110 -110
- package/lib/services/zone-drawer.service.d.ts +7 -7
- package/lib/services/zoneChange.service.d.ts +17 -17
- package/lib/types.service.d.ts +842 -842
- package/lib/validators/email.directive.d.ts +2 -2
- package/lib/validators/no-empty.directive.d.ts +2 -2
- package/lib/validators/number.directive.d.ts +2 -2
- package/lib/validators/text.directive.d.ts +2 -2
- package/package.json +2 -2
- package/public-api.d.ts +64 -64
- package/smarterplan-ngx-smarterplan-core.d.ts +5 -5
|
@@ -1,1587 +1,1587 @@
|
|
|
1
|
-
import { Inject, Injectable } from '@angular/core';
|
|
2
|
-
import { Subject } from 'rxjs';
|
|
3
|
-
//import * as serialijse from "serialijse";
|
|
4
|
-
import { getDistanceBetweenTwoPoints, poiTypeToString, wait, } from '../helpers.service';
|
|
5
|
-
import { MattertagData } from '../mattertagData';
|
|
6
|
-
import { FeatureType, MattertagActionMode, PoiType, SpModule, TagAction, ViewerInteractions, } from '../types.service';
|
|
7
|
-
import { Object3D } from 'three';
|
|
8
|
-
import { CameraMode } from '../types.service';
|
|
9
|
-
import { NestThermostat } from '../matterport-extensions/nest-thermostat/NestThermostat';
|
|
10
|
-
import { PlaneRenderer } from '../matterport-extensions/nest-thermostat/PlaneRenderer';
|
|
11
|
-
import { CanvasRenderer } from "../matterport-extensions/nest-thermostat/CanvasRenderer";
|
|
12
|
-
import { TvPlayer } from "../matterport-extensions/tv-player/TvPlayer";
|
|
13
|
-
import * as i0 from "@angular/core";
|
|
14
|
-
import * as i1 from "@angular/router";
|
|
15
|
-
import * as i2 from "./baseVisibility.service";
|
|
16
|
-
import * as i3 from "../config";
|
|
17
|
-
export class MatterportService {
|
|
18
|
-
constructor(config, router, activeRoute, visibilityService, ngZone) {
|
|
19
|
-
this.router = router;
|
|
20
|
-
this.activeRoute = activeRoute;
|
|
21
|
-
this.visibilityService = visibilityService;
|
|
22
|
-
this.ngZone = ngZone;
|
|
23
|
-
this.slots = []; //SlotNode[] = [];
|
|
24
|
-
this.nodes = [];
|
|
25
|
-
this.lastCameraPosition = { x: 0.0, y: 0.0, z: 0.0 };
|
|
26
|
-
this.cursorPositionButtonDisplayed = false;
|
|
27
|
-
// Measure mode
|
|
28
|
-
this.isMeasureModeOn = false;
|
|
29
|
-
this.interactionMode = ViewerInteractions.DEFAULT;
|
|
30
|
-
// List of created Mattertag IDs in the current viewer session
|
|
31
|
-
this.mattertagIDs = [];
|
|
32
|
-
// Dictionnary of MattertagID and its data (mattertagData)
|
|
33
|
-
this.dictionnaryTags = new Map();
|
|
34
|
-
this.dictionnaryObjects3D = new Map();
|
|
35
|
-
this.dictionnarySceneObjects3D = new Map();
|
|
36
|
-
this.lastMeasure = [];
|
|
37
|
-
this.distancesLastMeasure = [];
|
|
38
|
-
this.resolution = {
|
|
39
|
-
width: 500,
|
|
40
|
-
height: 600,
|
|
41
|
-
};
|
|
42
|
-
this.visibility = {
|
|
43
|
-
mattertags: false,
|
|
44
|
-
sweeps: true,
|
|
45
|
-
};
|
|
46
|
-
this.tagsAttachments = {};
|
|
47
|
-
this.currentSweep = new Subject();
|
|
48
|
-
//camera position with rotation
|
|
49
|
-
this.currentCameraPose = new Subject();
|
|
50
|
-
this.forbiddenSweeps = [];
|
|
51
|
-
this.tagAction = new Subject();
|
|
52
|
-
this.inTransitionMode = false;
|
|
53
|
-
this.inTransitionSweep = false;
|
|
54
|
-
this.noLightForObjects = true;
|
|
55
|
-
this.currentCameraMode = CameraMode.OUTSIDE;
|
|
56
|
-
this.onCameraModeChanged = new Subject();
|
|
57
|
-
this.onGoToTag = new Subject();
|
|
58
|
-
this.tagMessengerOn = false;
|
|
59
|
-
/**
|
|
60
|
-
* Actions on left click when positioning mattertag in visit
|
|
61
|
-
*/
|
|
62
|
-
this.pointerLeftClickHandler = () => {
|
|
63
|
-
if (this.mattertagToFollow) {
|
|
64
|
-
const mattertagData = this.dictionnaryTags.get(this.mattertagToFollow);
|
|
65
|
-
mattertagData.setPosition({ ...this.poseMatterport.position }); // copy!! not the reference
|
|
66
|
-
mattertagData.setNormal({ ...this.poseMatterport.normal }); // copy!! not the reference
|
|
67
|
-
this.dictionnaryTags.set(this.mattertagToFollow, mattertagData);
|
|
68
|
-
this.updateMatterTagContentForTagID(this.mattertagToFollow);
|
|
69
|
-
}
|
|
70
|
-
this.onValidatedMattertag();
|
|
71
|
-
};
|
|
72
|
-
this.pointerRightClickHandler = (e) => {
|
|
73
|
-
e.preventDefault();
|
|
74
|
-
this.cancelFollowingCursor();
|
|
75
|
-
alert('action cancelled');
|
|
76
|
-
};
|
|
77
|
-
this.pointerMiddleClickHandler = (e) => {
|
|
78
|
-
this.textDisplayCursorPositionPanel.innerHTML = `position:
|
|
79
|
-
${this.pointToString(this.poseMatterport.position)}\n
|
|
80
|
-
normal:
|
|
81
|
-
${this.pointToString(this.poseMatterport.normal)}\n
|
|
82
|
-
floorId:
|
|
83
|
-
${this.poseMatterport.floorId}`;
|
|
84
|
-
// this.textDisplayCursorPositionPanel.style.display = 'visible';
|
|
85
|
-
this.getCursorPositionButton.style.display = 'none';
|
|
86
|
-
};
|
|
87
|
-
this.config = config;
|
|
88
|
-
// TODO: only for dev!
|
|
89
|
-
if (!!this.getCursorPositionButton &&
|
|
90
|
-
(document.location.href.indexOf('dev') !== -1 || document.location.href.indexOf('localhost') !== -1)) {
|
|
91
|
-
this.intervalCursorPointerPosition = setInterval(() => {
|
|
92
|
-
if (!this.poseMatterport) {
|
|
93
|
-
return;
|
|
94
|
-
}
|
|
95
|
-
const nextShow = this.poseMatterport.time + 1000;
|
|
96
|
-
if (new Date().getTime() > nextShow) {
|
|
97
|
-
if (this.cursorPositionButtonDisplayed) {
|
|
98
|
-
return;
|
|
99
|
-
}
|
|
100
|
-
const size = {
|
|
101
|
-
w: this.container.clientWidth,
|
|
102
|
-
h: this.container.clientHeight,
|
|
103
|
-
};
|
|
104
|
-
const coord = this.sdk.Conversion.worldToScreen(this.poseMatterport.position, this.poseCamera, size);
|
|
105
|
-
this.getCursorPositionButton.style.left = `${coord.x - 25}px`;
|
|
106
|
-
this.getCursorPositionButton.style.top = `${coord.y - 22}px`;
|
|
107
|
-
this.getCursorPositionButton.style.display = 'block';
|
|
108
|
-
// this.textDisplayCursorPositionPanel.style.display = 'none';
|
|
109
|
-
this.cursorPositionButtonDisplayed = true;
|
|
110
|
-
}
|
|
111
|
-
}, 500);
|
|
112
|
-
}
|
|
113
|
-
}
|
|
114
|
-
get currentSpaceID() {
|
|
115
|
-
return this._currentSpaceID;
|
|
116
|
-
}
|
|
117
|
-
set currentSpaceID(value) {
|
|
118
|
-
this._currentSpaceID = value;
|
|
119
|
-
}
|
|
120
|
-
/**
|
|
121
|
-
* Initializes Matterport and all listeners/data
|
|
122
|
-
* @param tagService BaseTagService (to inject html)
|
|
123
|
-
* @param module SpModule (Museum, Immo) to subscribe only to needed functionnnalities
|
|
124
|
-
* @returns boolean
|
|
125
|
-
*/
|
|
126
|
-
async initSdk(tagService, module = SpModule.IMMO) {
|
|
127
|
-
if (this.sdk) {
|
|
128
|
-
// clean if sdk is running already
|
|
129
|
-
await this.action_delete_all_mattertags();
|
|
130
|
-
}
|
|
131
|
-
this.tagService = tagService;
|
|
132
|
-
this.SPModule = module;
|
|
133
|
-
return new Promise((resolve, reject) => {
|
|
134
|
-
// Retrieve DOM elements
|
|
135
|
-
this.pointerButton = document.querySelector('#viewer-pointer-trick');
|
|
136
|
-
this.container = document.querySelector('#viewer-module');
|
|
137
|
-
const iframe = document.querySelector('#viewer-module');
|
|
138
|
-
if (!iframe) {
|
|
139
|
-
return;
|
|
140
|
-
}
|
|
141
|
-
// Add listeners
|
|
142
|
-
if (this.pointerButton) {
|
|
143
|
-
this.pointerButton.addEventListener('click', this.pointerLeftClickHandler);
|
|
144
|
-
// cancel on right click
|
|
145
|
-
this.pointerButton.addEventListener('contextmenu', this.pointerRightClickHandler);
|
|
146
|
-
}
|
|
147
|
-
// Retrieve DOM elements
|
|
148
|
-
this.getCursorPositionButton = document.querySelector('#button');
|
|
149
|
-
this.textDisplayCursorPositionPanel = document.querySelector('#text');
|
|
150
|
-
if (this.getCursorPositionButton) {
|
|
151
|
-
// get position on Matterport "model" on middle click
|
|
152
|
-
this.getCursorPositionButton.addEventListener('click', this.pointerMiddleClickHandler);
|
|
153
|
-
}
|
|
154
|
-
// Load SDK
|
|
155
|
-
console.log('Loading Matterport SDK');
|
|
156
|
-
const showcaseWindow = iframe.contentWindow;
|
|
157
|
-
iframe.addEventListener('load', async function () {
|
|
158
|
-
try {
|
|
159
|
-
this.sdk = await showcaseWindow.MP_SDK.connect(showcaseWindow, 'qn9wsasuy5h2fzrbrn1nzr0id', '3.5');
|
|
160
|
-
}
|
|
161
|
-
catch (e) {
|
|
162
|
-
console.error(e);
|
|
163
|
-
return;
|
|
164
|
-
}
|
|
165
|
-
// Load Mattertag icons and custom subscriptions
|
|
166
|
-
switch (module) {
|
|
167
|
-
case SpModule.IMMO:
|
|
168
|
-
this.sdk.Asset.registerTexture('icon-ticket', this.config.my_config.icon_ticket);
|
|
169
|
-
this.sdk.Asset.registerTexture('icon-equipment', this.config.my_config.icon_equipment);
|
|
170
|
-
this.sdk.Asset.registerTexture('icon-measure', this.config.my_config.icon_measure);
|
|
171
|
-
this.sdk.Asset.registerTexture('icon-data', this.config.my_config.icon_data);
|
|
172
|
-
this.sdk.Asset.registerTexture('icon-object3d', this.config.my_config.icon_object3d);
|
|
173
|
-
this.sdk.Measurements.data.subscribe({
|
|
174
|
-
onAdded: function (index, item, collection) {
|
|
175
|
-
// console.log(
|
|
176
|
-
// "item added to the collection",
|
|
177
|
-
// index,
|
|
178
|
-
// item,
|
|
179
|
-
// );
|
|
180
|
-
// this.measurements[index] = item;
|
|
181
|
-
this.lastMeasure = item.points;
|
|
182
|
-
}.bind(this),
|
|
183
|
-
onCollectionUpdated: function (collection) {
|
|
184
|
-
// console.log('the entire up-to-date collection', collection);
|
|
185
|
-
this.getDistanceForLastMeasurement();
|
|
186
|
-
}.bind(this),
|
|
187
|
-
});
|
|
188
|
-
this.sdk.Measurements.mode.subscribe(function (measurementModeState) {
|
|
189
|
-
// measurement mode state has changed
|
|
190
|
-
this.isMeasureModeOn = measurementModeState.active;
|
|
191
|
-
// console.log('Is measurement mode currently active? ', measurementModeState.active);
|
|
192
|
-
}.bind(this));
|
|
193
|
-
break;
|
|
194
|
-
case SpModule.MUSEUM:
|
|
195
|
-
this.sdk.Asset.registerTexture('icon-data', this.config.my_config.icon_data);
|
|
196
|
-
break;
|
|
197
|
-
case SpModule.HOTEL:
|
|
198
|
-
this.sdk.Asset.registerTexture('icon_room_available', this.config.my_config.icon_room_available);
|
|
199
|
-
this.sdk.Asset.registerTexture('icon_room_unavailable', this.config.my_config.icon_room_unavailable);
|
|
200
|
-
this.sdk.Asset.registerTexture('icon_room_chosen', this.config.my_config.icon_room_chosen);
|
|
201
|
-
this.sdk.Asset.registerTexture('icon_room_viewed', this.config.my_config.icon_room_viewed);
|
|
202
|
-
break;
|
|
203
|
-
default:
|
|
204
|
-
break;
|
|
205
|
-
}
|
|
206
|
-
this.sdk.Asset.registerTexture('icon-position', this.config.my_config.icon_position);
|
|
207
|
-
// Current Room (used for getting bounding box and eventually lazy load mattertag or object3D inside current Room)
|
|
208
|
-
this.sdk.Room.current.subscribe((currentRoom) => {
|
|
209
|
-
console.log(`hello current Room ${JSON.stringify(currentRoom)}`);
|
|
210
|
-
});
|
|
211
|
-
// 3D position's pointer
|
|
212
|
-
this.sdk.Pointer.intersection.subscribe(function (intersection) {
|
|
213
|
-
this.poseMatterport = intersection;
|
|
214
|
-
this.poseMatterport.time = new Date().getTime();
|
|
215
|
-
if (!this.oldPoseMatterportPosition) {
|
|
216
|
-
this.oldPoseMatterportPosition = {
|
|
217
|
-
x: 0,
|
|
218
|
-
y: 0,
|
|
219
|
-
z: 0,
|
|
220
|
-
};
|
|
221
|
-
}
|
|
222
|
-
if (this.interactionMode !== ViewerInteractions.DEFAULT &&
|
|
223
|
-
this.mattertagToFollow) {
|
|
224
|
-
// follow the pointer and changes the position of the last tag
|
|
225
|
-
// (we are about to validate, but it exists in sdk already)
|
|
226
|
-
this.enable_following_tag(this.mattertagToFollow);
|
|
227
|
-
}
|
|
228
|
-
if (!!this.getCursorPositionButton && !!this.getCursorPositionButton.style &&
|
|
229
|
-
(document.URL.indexOf('https://dev.smarterplan.io') !== -1 || document.location.href.indexOf('localhost') !== -1)) {
|
|
230
|
-
this.getCursorPositionButton.style.display = 'none';
|
|
231
|
-
this.cursorPositionButtonDisplayed = false;
|
|
232
|
-
}
|
|
233
|
-
}.bind(this));
|
|
234
|
-
//Camera mode
|
|
235
|
-
this.sdk.Mode.current.subscribe((mode) => {
|
|
236
|
-
this.inTransitionSweep = false;
|
|
237
|
-
switch (mode) {
|
|
238
|
-
case 'mode.dollhouse':
|
|
239
|
-
this.currentCameraMode = CameraMode.DOLLHOUSE;
|
|
240
|
-
break;
|
|
241
|
-
case 'mode.floorplan':
|
|
242
|
-
this.currentCameraMode = CameraMode.FLOORPLAN;
|
|
243
|
-
break;
|
|
244
|
-
case 'mode.inside':
|
|
245
|
-
this.currentCameraMode = CameraMode.INSIDE;
|
|
246
|
-
break;
|
|
247
|
-
case 'mode.transitioning':
|
|
248
|
-
this.currentCameraMode = CameraMode.TRANSITIONING;
|
|
249
|
-
break;
|
|
250
|
-
default:
|
|
251
|
-
this.currentCameraMode = CameraMode.OUTSIDE;
|
|
252
|
-
}
|
|
253
|
-
this.onCameraModeChanged.next(this.currentCameraMode);
|
|
254
|
-
});
|
|
255
|
-
// Camera's viewpoint
|
|
256
|
-
this.sdk.Camera.pose.subscribe(function subscr(pose) {
|
|
257
|
-
this.poseCamera = pose;
|
|
258
|
-
this.currentCameraPose.next(pose);
|
|
259
|
-
// console.log('Current position is ', pose.position);
|
|
260
|
-
// console.log('Rotation angle is ', pose.rotation);
|
|
261
|
-
// console.log("Sweep UUID is", pose.sweep);
|
|
262
|
-
}.bind(this));
|
|
263
|
-
// subscribe to sweeps
|
|
264
|
-
this.sdk.Sweep.data.subscribe({
|
|
265
|
-
onCollectionUpdated: function subscr(collection) {
|
|
266
|
-
// console.log("Sweep collection updated");
|
|
267
|
-
this.sweeps = Object.keys(collection);
|
|
268
|
-
}.bind(this),
|
|
269
|
-
});
|
|
270
|
-
// subscribe to current sweep
|
|
271
|
-
this.sdk.Sweep.current.subscribe(function subscr(currentSweep) {
|
|
272
|
-
// Change to the current sweep has occurred.
|
|
273
|
-
if (currentSweep.sid === '') {
|
|
274
|
-
// console.log('Not currently stationed at a sweep position');
|
|
275
|
-
}
|
|
276
|
-
else {
|
|
277
|
-
// console.log("emmiting sweep", currentSweep.sid);
|
|
278
|
-
this.currentSweep.next(currentSweep.sid);
|
|
279
|
-
}
|
|
280
|
-
}.bind(this));
|
|
281
|
-
// Subscribe to Floor data
|
|
282
|
-
this.sdk.Floor.data.subscribe({
|
|
283
|
-
onCollectionUpdated: function upd(collection) {
|
|
284
|
-
this.floors = Object.values(collection);
|
|
285
|
-
}.bind(this),
|
|
286
|
-
});
|
|
287
|
-
// on tag click open details page
|
|
288
|
-
this.sdk.on('tag.click', (sid) => {
|
|
289
|
-
// get object of this tag
|
|
290
|
-
try {
|
|
291
|
-
const mattertagData = this.dictionnaryTags.get(sid);
|
|
292
|
-
if (!mattertagData) {
|
|
293
|
-
console.log('no mattertagData to display');
|
|
294
|
-
return;
|
|
295
|
-
}
|
|
296
|
-
const url = tagService.getUrlForSeeDetails(mattertagData.getObject(), mattertagData.getType());
|
|
297
|
-
if (url !== '') {
|
|
298
|
-
this.visibilityService.detailShowing.next(true);
|
|
299
|
-
this.ngZone.run(() => {
|
|
300
|
-
this.router.navigate([url]);
|
|
301
|
-
});
|
|
302
|
-
}
|
|
303
|
-
}
|
|
304
|
-
catch {
|
|
305
|
-
console.log('Cannot show details for tag');
|
|
306
|
-
}
|
|
307
|
-
});
|
|
308
|
-
// Pointer trick
|
|
309
|
-
// Create a div that will appear when the cursor is still
|
|
310
|
-
// It will intercept the click of the mouse before it trigerring the Matterport's tour navigation
|
|
311
|
-
this.timerPointer = setInterval(this.updatePointerTrick.bind(this), 50);
|
|
312
|
-
/**
|
|
313
|
-
* Transitions
|
|
314
|
-
*/
|
|
315
|
-
this.sdk.on('viewmode.changestart', function (to, from) {
|
|
316
|
-
// console.log('Starting to move to ' + to + ' from ' + from);
|
|
317
|
-
this.inTransitionMode = true;
|
|
318
|
-
}.bind(this));
|
|
319
|
-
this.sdk.on('viewmode.changeend', function (oldMode, newMode) {
|
|
320
|
-
// console.log('Ended to move to ' + newMode + ' from ' + oldMode);
|
|
321
|
-
if (newMode !== 'mode.transitioning') {
|
|
322
|
-
this.inTransitionMode = false;
|
|
323
|
-
}
|
|
324
|
-
}.bind(this));
|
|
325
|
-
this.sdk.on(this.sdk.Sweep.Event.ENTER, function (oldSweep, newSweep) {
|
|
326
|
-
// console.log('Leaving sweep ' + oldSweep);
|
|
327
|
-
// console.log('Entering sweep ' + newSweep);
|
|
328
|
-
this.inTransitionSweep = false;
|
|
329
|
-
this.displayAzimutalCrown();
|
|
330
|
-
}.bind(this));
|
|
331
|
-
this.sdk.on(this.sdk.Sweep.Event.EXIT, function (fromSweep, toSweep) {
|
|
332
|
-
// console.log('Leaving sweep ' + fromSweep);
|
|
333
|
-
// console.log('Transitioning to sweep ' + toSweep);
|
|
334
|
-
this.inTransitionSweep = true;
|
|
335
|
-
}.bind(this));
|
|
336
|
-
// TODO: get scene with getter instead!
|
|
337
|
-
const [sceneObject] = await this.sdk.Scene.createObjects(1);
|
|
338
|
-
const node = sceneObject.addNode();
|
|
339
|
-
node.start();
|
|
340
|
-
this.threeJSScene = node.obj3D.parent;
|
|
341
|
-
// await this.sdk.Scene.configure((renderer: any, three: any) => {
|
|
342
|
-
// renderer.physicallyCorrectLights = true;
|
|
343
|
-
// renderer.outputEncoding = three.sRGBEncoding;
|
|
344
|
-
// renderer.shadowMap.enabled = true;
|
|
345
|
-
// renderer.shadowMap.bias = 0.0001;
|
|
346
|
-
// renderer.shadowMap.type = three.PCFSoftShadowMap;
|
|
347
|
-
// });
|
|
348
|
-
// TODO: wait for MP ticket resolution before decomment these!
|
|
349
|
-
// Wait until Showcase is actually playing....
|
|
350
|
-
// this.sdk.Tag.data.subscribe({
|
|
351
|
-
// onAdded: async function (index, item, collection) {
|
|
352
|
-
//
|
|
353
|
-
// let thisOpacity = 0.2;
|
|
354
|
-
// this.sdk.Tag.editOpacity(index, thisOpacity);
|
|
355
|
-
//
|
|
356
|
-
// let source = null;
|
|
357
|
-
// try {
|
|
358
|
-
// source = await Promise.all([
|
|
359
|
-
// this.sdk.Sensor.createSource(this.sdk.Sensor.SourceType.SPHERE, {
|
|
360
|
-
// origin: item.anchorPosition,
|
|
361
|
-
// radius: Number(3),
|
|
362
|
-
// userData: {
|
|
363
|
-
// id: index + '-sphere-source',
|
|
364
|
-
// },
|
|
365
|
-
// })
|
|
366
|
-
// ]);
|
|
367
|
-
// } catch (e) {
|
|
368
|
-
// console.log('could not create Sphere sensor')
|
|
369
|
-
// console.error(e);
|
|
370
|
-
// }
|
|
371
|
-
// if (!source) {
|
|
372
|
-
// return;
|
|
373
|
-
// }
|
|
374
|
-
//
|
|
375
|
-
// let sensor = null;
|
|
376
|
-
// try {
|
|
377
|
-
// sensor = await this.sdk.Sensor.createSensor(this.sdk.Sensor.SensorType.CAMERA);
|
|
378
|
-
// } catch (e) {
|
|
379
|
-
// console.log('could not create Camera sensor')
|
|
380
|
-
// console.error(e);
|
|
381
|
-
// }
|
|
382
|
-
// if (!sensor) {
|
|
383
|
-
// return;
|
|
384
|
-
// }
|
|
385
|
-
//
|
|
386
|
-
// sensor.addSource(...source);
|
|
387
|
-
// sensor.readings.subscribe({
|
|
388
|
-
// onUpdated(source, reading) {
|
|
389
|
-
// console.log(thisOpacity);
|
|
390
|
-
// let oldOpacity = thisOpacity;
|
|
391
|
-
// if (reading.inRange) {
|
|
392
|
-
// thisOpacity = 1;
|
|
393
|
-
// console.log(index + ' is inRange');
|
|
394
|
-
// } else if (reading.inView) {
|
|
395
|
-
// console.log(index + ' is inView but not inRange');
|
|
396
|
-
// thisOpacity = 0.5;
|
|
397
|
-
// } else {
|
|
398
|
-
// thisOpacity = 0.2;
|
|
399
|
-
// console.log(index + ' is not inView or inRange');
|
|
400
|
-
// }
|
|
401
|
-
//
|
|
402
|
-
// this.sdk.Tag.editOpacity(index, thisOpacity);
|
|
403
|
-
// /*
|
|
404
|
-
// let inc = 0.01;
|
|
405
|
-
// if (oldOpacity > thisOpacity) {
|
|
406
|
-
// inc = -0.01;
|
|
407
|
-
// }
|
|
408
|
-
//
|
|
409
|
-
// for(var i = oldOpacity; i != thisOpacity; i=i+inc) {
|
|
410
|
-
// setTimeout(function() {
|
|
411
|
-
// mpSdk.Tag.editOpacity(index, i);
|
|
412
|
-
// console.log('Delay', i);
|
|
413
|
-
// },10);
|
|
414
|
-
// }
|
|
415
|
-
// */
|
|
416
|
-
// }
|
|
417
|
-
// });
|
|
418
|
-
// //sensor.showDebug(true);
|
|
419
|
-
// }.bind(this),
|
|
420
|
-
// onCollectionUpdated: function (collection) {
|
|
421
|
-
// console.log('Collection received. There are ', Object.keys(collection).length, ' Tags in the collection', collection);
|
|
422
|
-
// }
|
|
423
|
-
// });
|
|
424
|
-
resolve(true);
|
|
425
|
-
}.bind(this));
|
|
426
|
-
});
|
|
427
|
-
}
|
|
428
|
-
setLightingOff() {
|
|
429
|
-
this.noLightForObjects = true;
|
|
430
|
-
}
|
|
431
|
-
pointToString(point) {
|
|
432
|
-
var x = point.x.toFixed(3);
|
|
433
|
-
var y = point.y.toFixed(3);
|
|
434
|
-
var z = point.z.toFixed(3);
|
|
435
|
-
return `{ x: ${x}, y: ${y}, z: ${z} }`;
|
|
436
|
-
}
|
|
437
|
-
//
|
|
438
|
-
// ---------- Measurements related ----------
|
|
439
|
-
//
|
|
440
|
-
/**
|
|
441
|
-
* Callback after measurement is performed
|
|
442
|
-
*/
|
|
443
|
-
getDistanceForLastMeasurement() {
|
|
444
|
-
if (this.lastMeasure.length > 0) {
|
|
445
|
-
const numberPoints = this.lastMeasure.length;
|
|
446
|
-
this.distancesLastMeasure = [];
|
|
447
|
-
for (let index = 1; index < numberPoints; index += 1) {
|
|
448
|
-
const distance = getDistanceBetweenTwoPoints(this.lastMeasure[index - 1], this.lastMeasure[index]);
|
|
449
|
-
this.distancesLastMeasure.push(distance);
|
|
450
|
-
}
|
|
451
|
-
this.takeScreenShot().then((res) => {
|
|
452
|
-
this.router.navigate([`visit/${this.currentSpaceID}/add_measurement`]);
|
|
453
|
-
});
|
|
454
|
-
}
|
|
455
|
-
}
|
|
456
|
-
getLastDistances() {
|
|
457
|
-
return this.distancesLastMeasure;
|
|
458
|
-
}
|
|
459
|
-
/**
|
|
460
|
-
* Takes screenshot and saves base64 in lastScreenshotUri
|
|
461
|
-
* @returns Promise
|
|
462
|
-
*/
|
|
463
|
-
takeScreenShot() {
|
|
464
|
-
return this.sdk.Renderer.takeScreenShot(this.resolution, this.visibility).then(function (screenShotUri) {
|
|
465
|
-
// base64 string
|
|
466
|
-
this.lastScreenshotUri = screenShotUri;
|
|
467
|
-
return Promise.resolve();
|
|
468
|
-
}.bind(this));
|
|
469
|
-
}
|
|
470
|
-
getScreenShotUri() {
|
|
471
|
-
return this.lastScreenshotUri;
|
|
472
|
-
}
|
|
473
|
-
getLastMeasurement() {
|
|
474
|
-
const data = {
|
|
475
|
-
measure: this.lastMeasure,
|
|
476
|
-
sweep: this.poseCamera.sweep,
|
|
477
|
-
};
|
|
478
|
-
return data;
|
|
479
|
-
}
|
|
480
|
-
//
|
|
481
|
-
// ---------- Utils ----------
|
|
482
|
-
//
|
|
483
|
-
/**
|
|
484
|
-
* Styling of pointer
|
|
485
|
-
*/
|
|
486
|
-
updatePointerTrick() {
|
|
487
|
-
if (this.interactionMode !== ViewerInteractions.DEFAULT &&
|
|
488
|
-
this.mattertagToFollow &&
|
|
489
|
-
this.poseMatterport &&
|
|
490
|
-
this.getDistPosition(this.poseMatterport.position, this.oldPoseMatterportPosition) > 25) {
|
|
491
|
-
this.pointerButton.style.display = 'none';
|
|
492
|
-
const size = {
|
|
493
|
-
w: this.container.clientWidth,
|
|
494
|
-
h: this.container.clientHeight,
|
|
495
|
-
};
|
|
496
|
-
const coords = this.sdk.Conversion.worldToScreen(this.poseMatterport.position, this.poseCamera, size);
|
|
497
|
-
this.pointerButton.style.left = `${coords.x * Math.sign(coords.x) - 25}px`;
|
|
498
|
-
this.pointerButton.style.top = `${coords.y * Math.sign(coords.x) - 25}px`;
|
|
499
|
-
this.oldPoseMatterportPosition = {
|
|
500
|
-
...this.poseMatterport.position,
|
|
501
|
-
};
|
|
502
|
-
}
|
|
503
|
-
else {
|
|
504
|
-
this.pointerButton.style.display = 'block';
|
|
505
|
-
}
|
|
506
|
-
}
|
|
507
|
-
/**
|
|
508
|
-
* Realtime mattertag following the cursor
|
|
509
|
-
* @param mattertag string
|
|
510
|
-
*/
|
|
511
|
-
enable_following_tag(mattertag) {
|
|
512
|
-
this.sdk.Tag.editPosition(mattertag, {
|
|
513
|
-
anchorPosition: {
|
|
514
|
-
x: this.poseMatterport.position.x * 1,
|
|
515
|
-
y: this.poseMatterport.position.y * 1,
|
|
516
|
-
z: this.poseMatterport.position.z * 1,
|
|
517
|
-
},
|
|
518
|
-
stemVector: {
|
|
519
|
-
x: this.poseMatterport.normal.x * 0.3,
|
|
520
|
-
y: this.poseMatterport.normal.y * 0.3,
|
|
521
|
-
z: this.poseMatterport.normal.z * 0.3,
|
|
522
|
-
},
|
|
523
|
-
});
|
|
524
|
-
}
|
|
525
|
-
/**
|
|
526
|
-
* Get the distance betwween two 3D positions
|
|
527
|
-
* Used in order to see how much the cursor has moved from the previous position
|
|
528
|
-
* @param pos1
|
|
529
|
-
* @param pos2
|
|
530
|
-
*/
|
|
531
|
-
getDistPosition(pos1, pos2) {
|
|
532
|
-
const size = {
|
|
533
|
-
w: this.container.clientWidth,
|
|
534
|
-
h: this.container.clientHeight,
|
|
535
|
-
};
|
|
536
|
-
const coords1 = this.sdk.Conversion.worldToScreen(pos1, this.poseCamera, size);
|
|
537
|
-
const coords2 = this.sdk.Conversion.worldToScreen(pos2, this.poseCamera, size);
|
|
538
|
-
return Math.sqrt((coords1.x - coords2.x) ** 2 + (coords1.y - coords2.y) ** 2);
|
|
539
|
-
}
|
|
540
|
-
//
|
|
541
|
-
// ---------- Mattertag related ----------
|
|
542
|
-
//
|
|
543
|
-
/**
|
|
544
|
-
* Creates the Mattertag that will follow the cursor
|
|
545
|
-
* @param mattertagData MattertagData
|
|
546
|
-
*/
|
|
547
|
-
async addCursorMattertag(mattertagData) {
|
|
548
|
-
if (!this.poseMatterport)
|
|
549
|
-
return;
|
|
550
|
-
this.mattertagToFollow = await this.addMattertagToViewer(mattertagData);
|
|
551
|
-
console.log('following the tag', this.mattertagToFollow);
|
|
552
|
-
this.sdk.Tag.editIcon(this.mattertagToFollow, 'icon-position');
|
|
553
|
-
this.sdk.Tag.editOpacity(this.mattertagToFollow, 1);
|
|
554
|
-
}
|
|
555
|
-
/**
|
|
556
|
-
* Adds Mattertag to viewer for an existing object with coordinates (in mattertagData.poi)
|
|
557
|
-
* (position, injected html, set icon)
|
|
558
|
-
* @param mattertagData
|
|
559
|
-
* returns mattertagID
|
|
560
|
-
*/
|
|
561
|
-
async addMattertagToViewer(mattertagData) {
|
|
562
|
-
let sidList;
|
|
563
|
-
if (!this.sdk) {
|
|
564
|
-
return null;
|
|
565
|
-
}
|
|
566
|
-
try {
|
|
567
|
-
sidList = await this.sdk.Tag.add(mattertagData.getData());
|
|
568
|
-
}
|
|
569
|
-
catch (error) {
|
|
570
|
-
console.log('Tag does not belong to the visit', error, mattertagData.getData());
|
|
571
|
-
}
|
|
572
|
-
if (sidList) {
|
|
573
|
-
const mattertagID = sidList[0];
|
|
574
|
-
// console.log("added tag", mattertagData.getType(), mattertagID);
|
|
575
|
-
this.mattertagIDs.push(mattertagID);
|
|
576
|
-
this.dictionnaryTags.set(mattertagID, mattertagData);
|
|
577
|
-
return mattertagID;
|
|
578
|
-
}
|
|
579
|
-
return null;
|
|
580
|
-
}
|
|
581
|
-
/**
|
|
582
|
-
* Actions when position of mattertag is validated by left click
|
|
583
|
-
*/
|
|
584
|
-
onValidatedMattertag() {
|
|
585
|
-
this.setInteractionMode(ViewerInteractions.DEFAULT);
|
|
586
|
-
const lastTag = this.getLastTag();
|
|
587
|
-
if (lastTag) {
|
|
588
|
-
const callbackMode = this.dictionnaryTags
|
|
589
|
-
.get(lastTag)
|
|
590
|
-
.getCallbackActionMode();
|
|
591
|
-
this.callbackAfterMattertagValidation(callbackMode);
|
|
592
|
-
}
|
|
593
|
-
}
|
|
594
|
-
/**
|
|
595
|
-
* Registers new icon (path to image) and set its for Mattertag
|
|
596
|
-
* @param mattertagID string
|
|
597
|
-
* @param iconPath string
|
|
598
|
-
*/
|
|
599
|
-
async addNewIconAndSetForTag(mattertagID, iconPath) {
|
|
600
|
-
await this.sdk.Asset.registerTexture(`icon_${mattertagID}`, iconPath);
|
|
601
|
-
this.sdk.Tag.editIcon(mattertagID, `icon_${mattertagID}`);
|
|
602
|
-
}
|
|
603
|
-
/**
|
|
604
|
-
* Changes icon of Mattertag (the iconName should be registered = one of default ones)
|
|
605
|
-
* @param mattertagID string
|
|
606
|
-
* @param iconName string
|
|
607
|
-
*/
|
|
608
|
-
async setRegistredIconForTag(mattertagID, iconName) {
|
|
609
|
-
try {
|
|
610
|
-
this.sdk.Tag.editIcon(mattertagID, iconName);
|
|
611
|
-
}
|
|
612
|
-
catch (e) {
|
|
613
|
-
console.log('could not edit Icon with name ', iconName, '. Is it registered?');
|
|
614
|
-
}
|
|
615
|
-
}
|
|
616
|
-
/**
|
|
617
|
-
* Sets default icon for a tag (registered in initSdk) OR uses tagIcon from POI (available from MattertagData)
|
|
618
|
-
* @param mattertagID string
|
|
619
|
-
* @param mattertagData MattertagData
|
|
620
|
-
* @returns
|
|
621
|
-
*/
|
|
622
|
-
async setTagIconAndOpacity(mattertagID, mattertagData) {
|
|
623
|
-
if (this.SPModule === SpModule.HOTEL) {
|
|
624
|
-
const room = mattertagData.getObject();
|
|
625
|
-
let iconName = `icon_room`;
|
|
626
|
-
if (room.available) {
|
|
627
|
-
iconName = `${iconName}_available`;
|
|
628
|
-
}
|
|
629
|
-
else {
|
|
630
|
-
iconName = `${iconName}_unavailable`;
|
|
631
|
-
}
|
|
632
|
-
this.sdk.Tag.editIcon(mattertagID, iconName);
|
|
633
|
-
return;
|
|
634
|
-
}
|
|
635
|
-
const stringTagType = poiTypeToString(mattertagData.getType());
|
|
636
|
-
let opacity = this.config.my_config.DEFAULT_OPACITY_TAG;
|
|
637
|
-
let iconName = `icon-${stringTagType}`;
|
|
638
|
-
const poi = mattertagData.getPoi();
|
|
639
|
-
if (poi && poi.tagIcon) {
|
|
640
|
-
const tagIcon = JSON.parse(poi.tagIcon);
|
|
641
|
-
if (mattertagData.getType() === PoiType.DATA) {
|
|
642
|
-
const feature = mattertagData.getObject();
|
|
643
|
-
if (feature.type === FeatureType.INDICATOR_TEMP) {
|
|
644
|
-
tagIcon.src = this.tagService.getIconTagImageForFeature(feature, poi);
|
|
645
|
-
}
|
|
646
|
-
}
|
|
647
|
-
if (tagIcon.src) {
|
|
648
|
-
const pathSigned = await this.tagService.getSignedTagIconSource(tagIcon.src);
|
|
649
|
-
if (pathSigned) {
|
|
650
|
-
try {
|
|
651
|
-
iconName = `icon-${stringTagType}-${mattertagID}-${mattertagData.customIconIndex}`;
|
|
652
|
-
await this.sdk.Asset.registerTexture(iconName, pathSigned);
|
|
653
|
-
mattertagData.customIconIndex += 1;
|
|
654
|
-
}
|
|
655
|
-
catch {
|
|
656
|
-
console.log('error registerIcon');
|
|
657
|
-
}
|
|
658
|
-
}
|
|
659
|
-
}
|
|
660
|
-
if (tagIcon.opacity) {
|
|
661
|
-
opacity = tagIcon.opacity;
|
|
662
|
-
}
|
|
663
|
-
}
|
|
664
|
-
try {
|
|
665
|
-
this.sdk.Tag.editIcon(mattertagID, iconName);
|
|
666
|
-
this.sdk.Tag.editOpacity(mattertagID, opacity);
|
|
667
|
-
}
|
|
668
|
-
catch (error) {
|
|
669
|
-
console.log('Error editIcon or opacity', error);
|
|
670
|
-
}
|
|
671
|
-
}
|
|
672
|
-
/**
|
|
673
|
-
* Moves viewer to last tag created
|
|
674
|
-
*/
|
|
675
|
-
goToLastTag() {
|
|
676
|
-
setTimeout(() => {
|
|
677
|
-
const lastTag = this.getLastTag();
|
|
678
|
-
this.goToTag(lastTag);
|
|
679
|
-
}, 2000);
|
|
680
|
-
}
|
|
681
|
-
/**
|
|
682
|
-
* Moves viewer to Mattertag with ID provided
|
|
683
|
-
* @param mattertagID string
|
|
684
|
-
* @returns
|
|
685
|
-
*/
|
|
686
|
-
async goToTag(mattertagID) {
|
|
687
|
-
if (mattertagID === '')
|
|
688
|
-
return;
|
|
689
|
-
try {
|
|
690
|
-
this.onGoToTag.next(mattertagID);
|
|
691
|
-
await this.sdk.Sweep.current.waitUntil((currentSweep) => currentSweep !== '');
|
|
692
|
-
await this.sdk.Mattertag.navigateToTag(mattertagID, this.sdk.Mattertag.Transition.FLY);
|
|
693
|
-
}
|
|
694
|
-
catch (error) {
|
|
695
|
-
console.log('cannot navigate to tag', error);
|
|
696
|
-
}
|
|
697
|
-
}
|
|
698
|
-
/**
|
|
699
|
-
* Updates content of Mattertag with mattertagID (billboard, injected html, tag icon)
|
|
700
|
-
* @param mattertagID string
|
|
701
|
-
* @param object Ticket, Equipment, Feature, etc
|
|
702
|
-
* @param tagType PoiType
|
|
703
|
-
*/
|
|
704
|
-
async updateMatterTagContentForTagID(mattertagID, object = null, tagType = null) {
|
|
705
|
-
this.sdk.Tag.editBillboard(mattertagID, this.dictionnaryTags.get(mattertagID).getData());
|
|
706
|
-
if (object && tagType) {
|
|
707
|
-
await this.injectHtmlInTag(tagType, object, mattertagID);
|
|
708
|
-
}
|
|
709
|
-
await this.setTagIconAndOpacity(mattertagID, this.dictionnaryTags.get(mattertagID));
|
|
710
|
-
}
|
|
711
|
-
/**
|
|
712
|
-
* Updates injected html for Mattertag
|
|
713
|
-
* @param mattertagID string
|
|
714
|
-
* @param object Ticket, Equipment, Feature, etc
|
|
715
|
-
* @param tagType PoiType
|
|
716
|
-
*/
|
|
717
|
-
async updateInjectedHtmlForTagID(mattertagID, object, tagType) {
|
|
718
|
-
await this.injectHtmlInTag(tagType, object, mattertagID);
|
|
719
|
-
}
|
|
720
|
-
/**
|
|
721
|
-
* Deletes Mattertag from Viewer by its ID
|
|
722
|
-
* @param mattertagID string
|
|
723
|
-
*/
|
|
724
|
-
deleteMattertagFromId(mattertagID) {
|
|
725
|
-
this.sdk.Tag.remove(mattertagID);
|
|
726
|
-
this.dictionnaryTags.delete(mattertagID);
|
|
727
|
-
}
|
|
728
|
-
/**
|
|
729
|
-
* Deletes latest created mattertag
|
|
730
|
-
*/
|
|
731
|
-
deleteLastMattertag() {
|
|
732
|
-
const mattertagID = this.mattertagIDs.pop();
|
|
733
|
-
if (mattertagID) {
|
|
734
|
-
this.deleteMattertagFromId(mattertagID);
|
|
735
|
-
}
|
|
736
|
-
}
|
|
737
|
-
/**
|
|
738
|
-
* Legacy: used to be called action_add_mattertag_from_POI
|
|
739
|
-
* Adds and configures Mattertag for an object (Ticket, Equipment, Feature, etc) that corresponds to POI (coordinates, tagIcon)
|
|
740
|
-
* @param tagType PoiType
|
|
741
|
-
* @param object Ticket, Equipment, Feature...
|
|
742
|
-
* @param poi POI
|
|
743
|
-
* @returns
|
|
744
|
-
*/
|
|
745
|
-
async createMattertagFromPOI(tagType, object, poi) {
|
|
746
|
-
// check if tag exists already
|
|
747
|
-
const { tag, sweep } = this.getTagFromElementId(object.id);
|
|
748
|
-
if (tag) {
|
|
749
|
-
// console.log("tag exists", object)
|
|
750
|
-
return;
|
|
751
|
-
}
|
|
752
|
-
const mattertagData = new MattertagData(tagType);
|
|
753
|
-
mattertagData.setObject(object, tagType);
|
|
754
|
-
if (poi.coordinate) {
|
|
755
|
-
mattertagData.setPosition(JSON.parse(poi.coordinate));
|
|
756
|
-
}
|
|
757
|
-
if (poi.metadata) {
|
|
758
|
-
const tagMetadata = JSON.parse(poi.metadata);
|
|
759
|
-
if (tagMetadata.normal) {
|
|
760
|
-
mattertagData.setNormal(tagMetadata.normal);
|
|
761
|
-
}
|
|
762
|
-
else {
|
|
763
|
-
mattertagData.setNormal({ x: 0, y: -0.15, z: 0 });
|
|
764
|
-
}
|
|
765
|
-
}
|
|
766
|
-
if (poi.matterportSweepID) {
|
|
767
|
-
mattertagData.setSweepID(poi.matterportSweepID);
|
|
768
|
-
}
|
|
769
|
-
mattertagData.setPoi(poi);
|
|
770
|
-
const createdTagID = await this.addMattertagToViewer(mattertagData);
|
|
771
|
-
if (createdTagID && this.sdk) {
|
|
772
|
-
await this.setTagIconAndOpacity(createdTagID, mattertagData);
|
|
773
|
-
await this.injectHtmlInTag(tagType, object, createdTagID);
|
|
774
|
-
}
|
|
775
|
-
}
|
|
776
|
-
/**
|
|
777
|
-
* Inject custom HTML as Mattertag content
|
|
778
|
-
* @param tagType PoiType
|
|
779
|
-
* @param object Ticket, Equipment, Feature etc
|
|
780
|
-
* @param tagID string
|
|
781
|
-
*/
|
|
782
|
-
async injectHtmlInTag(tagType, object, tagID) {
|
|
783
|
-
let html = await this.tagService.getHtmlToInject(tagType, object);
|
|
784
|
-
if (html !== '' && this.sdk) {
|
|
785
|
-
const scriptTag = this.tagService.getScriptForTag(object, tagType);
|
|
786
|
-
html += `${scriptTag}`;
|
|
787
|
-
// create and register the sandbox
|
|
788
|
-
const [sandboxId, messenger] = await this.sdk.Tag.registerSandbox(html);
|
|
789
|
-
// detach previous sandbox from a tag
|
|
790
|
-
let attachmentID = this.tagsAttachments[tagID];
|
|
791
|
-
if (attachmentID) {
|
|
792
|
-
this.sdk.Tag.detach(tagID, attachmentID);
|
|
793
|
-
}
|
|
794
|
-
this.tagsAttachments[tagID] = sandboxId;
|
|
795
|
-
// attach the sandbox to a tag
|
|
796
|
-
this.sdk.Tag.attach(tagID, sandboxId);
|
|
797
|
-
// receive data from the sandbox
|
|
798
|
-
// tagMessengerOn allows to go here only once
|
|
799
|
-
if (!this.tagMessengerOn) {
|
|
800
|
-
this.tagMessengerOn = true;
|
|
801
|
-
const imageClick = (featureID) => {
|
|
802
|
-
// console.log("image click handler", featureID);
|
|
803
|
-
this.tagService.onActionImageClick(featureID);
|
|
804
|
-
};
|
|
805
|
-
const audioClick = (audioCommentID) => {
|
|
806
|
-
// console.log("audio click handler", audioCommentID);
|
|
807
|
-
this.tagService.onActionAudioClick(audioCommentID);
|
|
808
|
-
};
|
|
809
|
-
const videoClick = (url) => {
|
|
810
|
-
this.tagService.onActionVideoClick(url);
|
|
811
|
-
};
|
|
812
|
-
messenger.on(TagAction.DETAIL_CLICK, this.tagService.onActionDetailClick.bind(this.tagService));
|
|
813
|
-
messenger.on(TagAction.TICKET_CLICK, this.tagService.onActionDetailClick.bind(this.tagService));
|
|
814
|
-
messenger.on(TagAction.AUDIO_CLICK, audioClick);
|
|
815
|
-
messenger.on(TagAction.IMAGE_CLICK, imageClick);
|
|
816
|
-
messenger.on(TagAction.VIDEO_CLICK, videoClick);
|
|
817
|
-
messenger.on(TagAction.DOC_CLICK, this.tagService.onActionDocClick.bind(this.tagService));
|
|
818
|
-
messenger.on(TagAction.YOUTUBE_CLICK, this.tagService.onActionYoutubeClick.bind(this.tagService));
|
|
819
|
-
}
|
|
820
|
-
}
|
|
821
|
-
else {
|
|
822
|
-
// if html is empty (case of EMBED content), we edit billboard for Feature and attach new content
|
|
823
|
-
const { comment, tagDescription } = this.tagService.getBillboardMediaToEmbed(object);
|
|
824
|
-
if (comment) {
|
|
825
|
-
// register the media
|
|
826
|
-
const [attachmentID] = await this.sdk.Tag.registerAttachment(comment.externalLink);
|
|
827
|
-
// attach
|
|
828
|
-
this.sdk.Tag.attach(tagID, attachmentID);
|
|
829
|
-
this.sdk.Tag.editBillboard(tagID, {
|
|
830
|
-
label: object.title,
|
|
831
|
-
description: tagDescription,
|
|
832
|
-
});
|
|
833
|
-
}
|
|
834
|
-
}
|
|
835
|
-
}
|
|
836
|
-
async action_delete_all_mattertags() {
|
|
837
|
-
await this.sdk.Tag.remove(...this.mattertagIDs);
|
|
838
|
-
this.mattertagIDs = [];
|
|
839
|
-
this.dictionnaryTags.clear();
|
|
840
|
-
}
|
|
841
|
-
/**
|
|
842
|
-
* Deletes Mattertag from visit associated with object ID (ticketID, etc)
|
|
843
|
-
* @param elementID string
|
|
844
|
-
*/
|
|
845
|
-
async deleteMattertagForObject(elementID) {
|
|
846
|
-
const matterTagID = this.getTagFromElementId(elementID).tag;
|
|
847
|
-
if (matterTagID) {
|
|
848
|
-
try {
|
|
849
|
-
await this.sdk.Tag.remove(matterTagID);
|
|
850
|
-
this.dictionnaryTags.delete(matterTagID);
|
|
851
|
-
const index = this.mattertagIDs.indexOf(matterTagID);
|
|
852
|
-
this.mattertagIDs.splice(index, 1);
|
|
853
|
-
}
|
|
854
|
-
catch (error) {
|
|
855
|
-
console.log('Cannot delete tag', matterTagID, error);
|
|
856
|
-
}
|
|
857
|
-
}
|
|
858
|
-
}
|
|
859
|
-
/**
|
|
860
|
-
* uuid from threejs
|
|
861
|
-
* @param uuid
|
|
862
|
-
*/
|
|
863
|
-
async deleteObject3D(uuid) {
|
|
864
|
-
this.dictionnaryObjects3D.get(uuid).obj3D.visible = false;
|
|
865
|
-
}
|
|
866
|
-
getObject3DModelNodeFromDictionnary(uuid) {
|
|
867
|
-
let obj = this.dictionnaryObjects3D.get(uuid);
|
|
868
|
-
if (!obj) {
|
|
869
|
-
console.log("weird thing again");
|
|
870
|
-
return null;
|
|
871
|
-
}
|
|
872
|
-
//might break things or fix everything ..?
|
|
873
|
-
if (obj.obj3D.uuid != uuid) {
|
|
874
|
-
console.log("we have THE problem");
|
|
875
|
-
obj.obj3D.uuid = uuid; //not enugh to fix the pb
|
|
876
|
-
}
|
|
877
|
-
return obj;
|
|
878
|
-
}
|
|
879
|
-
/**
|
|
880
|
-
* Creates MattertagData and start repositioning (creates temporary mattertag that follows the cursor)
|
|
881
|
-
* @param poiType PoiType
|
|
882
|
-
* @param element Ticket, Equipment, Feature, etc
|
|
883
|
-
*/
|
|
884
|
-
async addMattertagWhenRepositioning(poiType, element) {
|
|
885
|
-
const mattertagData = new MattertagData(poiType);
|
|
886
|
-
// set the coordinates of the existing tag
|
|
887
|
-
const [poi] = element.pois.items;
|
|
888
|
-
if (poi && poi.coordinate) {
|
|
889
|
-
mattertagData.setPosition(JSON.parse(poi.coordinate));
|
|
890
|
-
mattertagData.setPoi(poi); // to keep custom tagIcon and opacity
|
|
891
|
-
}
|
|
892
|
-
mattertagData.setSweepID(this.poseCamera.sweep);
|
|
893
|
-
mattertagData.setRotation(this.poseCamera.rotation);
|
|
894
|
-
mattertagData.setObject(element, poiType);
|
|
895
|
-
this.mattertagToFollow = await this.addMattertagToViewer(mattertagData);
|
|
896
|
-
this.setInteractionMode(ViewerInteractions.POSITIONING);
|
|
897
|
-
await this.addCursorMattertag(mattertagData);
|
|
898
|
-
}
|
|
899
|
-
/**
|
|
900
|
-
* Creates MattertagData and mattertag that follows the cursor when choosing position for a new object
|
|
901
|
-
* @param poiType
|
|
902
|
-
*/
|
|
903
|
-
async addMattertagWhenAdding(poiType) {
|
|
904
|
-
const mattertagData = new MattertagData(poiType);
|
|
905
|
-
mattertagData.setSweepID(this.poseCamera.sweep);
|
|
906
|
-
mattertagData.setRotation(this.poseCamera.rotation);
|
|
907
|
-
this.setInteractionMode(ViewerInteractions.ADDING);
|
|
908
|
-
await this.addCursorMattertag(mattertagData);
|
|
909
|
-
}
|
|
910
|
-
/**
|
|
911
|
-
* Cancels following of cursor (meaning deleting last Mattertag)
|
|
912
|
-
*/
|
|
913
|
-
cancelFollowingCursor() {
|
|
914
|
-
this.deleteLastMattertag();
|
|
915
|
-
this.setInteractionMode(ViewerInteractions.DEFAULT);
|
|
916
|
-
}
|
|
917
|
-
setLastObject3D(lastObject3D) {
|
|
918
|
-
this.lastObject3D = lastObject3D;
|
|
919
|
-
}
|
|
920
|
-
getLastObject3D() {
|
|
921
|
-
return this.lastObject3D;
|
|
922
|
-
}
|
|
923
|
-
/**
|
|
924
|
-
* Performs callback after mattertag position was validated (creation of object or repositioning)
|
|
925
|
-
* @param mode MattertagActionMode
|
|
926
|
-
*/
|
|
927
|
-
callbackAfterMattertagValidation(mode) {
|
|
928
|
-
this.visibilityService.detailShowing.next(true);
|
|
929
|
-
const lastTag = this.getLastTag();
|
|
930
|
-
switch (mode) {
|
|
931
|
-
case MattertagActionMode.ADD_EQUIPMENT:
|
|
932
|
-
this.goToLastTag();
|
|
933
|
-
this.router.navigate([`visit/${this.currentSpaceID}/add_equip`]);
|
|
934
|
-
break;
|
|
935
|
-
case MattertagActionMode.ADD_TICKET:
|
|
936
|
-
this.goToLastTag();
|
|
937
|
-
this.router.navigate([`visit/${this.currentSpaceID}/add_ticket`]);
|
|
938
|
-
break;
|
|
939
|
-
case MattertagActionMode.ADD_OBJECT3D:
|
|
940
|
-
this.goToLastTag();
|
|
941
|
-
this.router.navigate([`visit/${this.currentSpaceID}/add_object3d`]);
|
|
942
|
-
break;
|
|
943
|
-
case MattertagActionMode.ADD_DATA:
|
|
944
|
-
this.goToLastTag();
|
|
945
|
-
let url;
|
|
946
|
-
if (this.router.url.includes('?')) {
|
|
947
|
-
url = this.router.url.substring(0, this.router.url.indexOf('?'));
|
|
948
|
-
}
|
|
949
|
-
else {
|
|
950
|
-
url = this.router.url;
|
|
951
|
-
}
|
|
952
|
-
this.router.navigate([`${url}/add_feature`]);
|
|
953
|
-
break;
|
|
954
|
-
case MattertagActionMode.ADD_DESK:
|
|
955
|
-
this.goToLastTag();
|
|
956
|
-
this.router.navigate([`visit/${this.currentSpaceID}/add_feature`], {
|
|
957
|
-
queryParams: { isDesk: true },
|
|
958
|
-
});
|
|
959
|
-
break;
|
|
960
|
-
case MattertagActionMode.ADD_ROOM:
|
|
961
|
-
this.goToLastTag();
|
|
962
|
-
this.router.navigate([`visit/${this.currentSpaceID}/add_room`]);
|
|
963
|
-
break;
|
|
964
|
-
case MattertagActionMode.POSITION_OBJECT3D:
|
|
965
|
-
if (lastTag) {
|
|
966
|
-
// DO Nothing (routing), just
|
|
967
|
-
this.router.navigate([`visit/${this.currentSpaceID}/object3d/${this.lastObject3D.id}`], { queryParams: { positioning: true } });
|
|
968
|
-
}
|
|
969
|
-
break;
|
|
970
|
-
case MattertagActionMode.POSITION_TICKET:
|
|
971
|
-
if (lastTag) {
|
|
972
|
-
this.router.navigate([
|
|
973
|
-
`visit/${this.currentSpaceID}/detail/${this.dictionnaryTags.get(lastTag).elementID}`,
|
|
974
|
-
], { queryParams: { positioning: true } });
|
|
975
|
-
}
|
|
976
|
-
break;
|
|
977
|
-
case MattertagActionMode.POSITION_EQUIPMENT:
|
|
978
|
-
if (lastTag) {
|
|
979
|
-
this.router.navigate([
|
|
980
|
-
`visit/${this.currentSpaceID}/equip/${this.dictionnaryTags.get(lastTag).elementID}`,
|
|
981
|
-
], { queryParams: { positioning: true } });
|
|
982
|
-
}
|
|
983
|
-
break;
|
|
984
|
-
case MattertagActionMode.POSITION_DATA:
|
|
985
|
-
if (lastTag) {
|
|
986
|
-
this.router.navigate([], {
|
|
987
|
-
relativeTo: this.activeRoute,
|
|
988
|
-
queryParams: { positioning: true }, queryParamsHandling: "merge"
|
|
989
|
-
});
|
|
990
|
-
}
|
|
991
|
-
break;
|
|
992
|
-
case MattertagActionMode.POSITION_ROOM:
|
|
993
|
-
if (lastTag) {
|
|
994
|
-
this.router.navigate([], {
|
|
995
|
-
relativeTo: this.activeRoute,
|
|
996
|
-
queryParams: { positioning: true }, queryParamsHandling: "merge"
|
|
997
|
-
});
|
|
998
|
-
}
|
|
999
|
-
break;
|
|
1000
|
-
}
|
|
1001
|
-
}
|
|
1002
|
-
/**
|
|
1003
|
-
* Fully updates existing Mattertag content with data of object (Ticket, Equipment, Desk)
|
|
1004
|
-
* @param mattertagID string
|
|
1005
|
-
* @param object Ticket, Equipment, Feature, Desk
|
|
1006
|
-
* @param poiType PoiType
|
|
1007
|
-
* @param poi POI
|
|
1008
|
-
*/
|
|
1009
|
-
async setObjectAndPoiInTag(mattertagID, object, poiType, poi = null) {
|
|
1010
|
-
this.dictionnaryTags.get(mattertagID).setObject(object, poiType);
|
|
1011
|
-
if (poi) {
|
|
1012
|
-
this.dictionnaryTags.get(mattertagID).setPoi(poi);
|
|
1013
|
-
}
|
|
1014
|
-
this.dictionnaryTags.get(mattertagID).elementID = object.id;
|
|
1015
|
-
try {
|
|
1016
|
-
await this.updateMatterTagContentForTagID(mattertagID, object, poiType);
|
|
1017
|
-
// TODO: fix this
|
|
1018
|
-
await this.updateMatterTagPosInSdkViewer(mattertagID, object, poiType, poi);
|
|
1019
|
-
}
|
|
1020
|
-
catch (e) {
|
|
1021
|
-
console.log(`error in setObjectAndPoiInTag: ${e}`);
|
|
1022
|
-
}
|
|
1023
|
-
}
|
|
1024
|
-
async updateMatterTagPosInSdkViewer(mattertagID, object, poiType, poi = null) {
|
|
1025
|
-
const IndexToUpdate = object.pois.items.findIndex((u) => u.id === poi.id);
|
|
1026
|
-
object.pois.items[IndexToUpdate] = poi;
|
|
1027
|
-
const { x, y, z } = JSON.parse(poi.coordinate);
|
|
1028
|
-
const newPosition = {
|
|
1029
|
-
anchorPosition: {
|
|
1030
|
-
x: x,
|
|
1031
|
-
y: y,
|
|
1032
|
-
z: z,
|
|
1033
|
-
},
|
|
1034
|
-
stemVector: {
|
|
1035
|
-
// make the Tag stick straight up and make it 0.30 meters (~1 foot) tall
|
|
1036
|
-
x: 0,
|
|
1037
|
-
y: 0.3,
|
|
1038
|
-
z: 0,
|
|
1039
|
-
},
|
|
1040
|
-
};
|
|
1041
|
-
this.sdk.Mattertag.editPosition(mattertagID, newPosition);
|
|
1042
|
-
this.sdk.Tag.editPosition(mattertagID, newPosition);
|
|
1043
|
-
this.dictionnaryTags
|
|
1044
|
-
.get(mattertagID)
|
|
1045
|
-
.setPosition(newPosition.anchorPosition);
|
|
1046
|
-
for (let tagId of this.mattertagIDs) {
|
|
1047
|
-
const currentTag = this.dictionnaryTags.get(tagId);
|
|
1048
|
-
if (!currentTag) {
|
|
1049
|
-
continue;
|
|
1050
|
-
}
|
|
1051
|
-
if (currentTag.elementID === object.id && tagId !== mattertagID) {
|
|
1052
|
-
this.dictionnaryTags.delete(tagId);
|
|
1053
|
-
this.sdk.Tag.remove(tagId);
|
|
1054
|
-
this.sdk.Mattertag.remove(tagId);
|
|
1055
|
-
break;
|
|
1056
|
-
}
|
|
1057
|
-
}
|
|
1058
|
-
}
|
|
1059
|
-
/**
|
|
1060
|
-
* Gets matterTagID and its sweep for an object ID (ticket ID, etc)
|
|
1061
|
-
* @param elementID string
|
|
1062
|
-
* @returns {tag: string | null, sweep: string | null}
|
|
1063
|
-
*/
|
|
1064
|
-
getTagFromElementId(elementID) {
|
|
1065
|
-
let tagID = null;
|
|
1066
|
-
let sweepID = null;
|
|
1067
|
-
for (let [mattertagID, mattertagData] of this.dictionnaryTags) {
|
|
1068
|
-
if (mattertagData.elementID === elementID) {
|
|
1069
|
-
tagID = mattertagID;
|
|
1070
|
-
const sweep = mattertagData.getSweepID();
|
|
1071
|
-
if (sweep && this.sweeps && this.sweeps.includes(sweep)) {
|
|
1072
|
-
sweepID = sweep;
|
|
1073
|
-
}
|
|
1074
|
-
}
|
|
1075
|
-
}
|
|
1076
|
-
return { tag: tagID, sweep: sweepID };
|
|
1077
|
-
}
|
|
1078
|
-
/**
|
|
1079
|
-
* Gets latest tag created in the visit (when following the cursor to position)
|
|
1080
|
-
* @returns string mattertagID
|
|
1081
|
-
*/
|
|
1082
|
-
getLastTag() {
|
|
1083
|
-
if (this.mattertagToFollow) {
|
|
1084
|
-
return this.mattertagToFollow;
|
|
1085
|
-
}
|
|
1086
|
-
else {
|
|
1087
|
-
return this.mattertagIDs.length === 0
|
|
1088
|
-
? null
|
|
1089
|
-
: this.mattertagIDs[this.mattertagIDs.length - 1];
|
|
1090
|
-
}
|
|
1091
|
-
}
|
|
1092
|
-
/**
|
|
1093
|
-
* Gets MattertagData for Mattertag (replaces getTagDataFromId)
|
|
1094
|
-
* @param mattertagID
|
|
1095
|
-
* @returns MattertagData
|
|
1096
|
-
*/
|
|
1097
|
-
getMatterTagDataForMattertag(mattertagID) {
|
|
1098
|
-
try {
|
|
1099
|
-
return this.dictionnaryTags.get(mattertagID);
|
|
1100
|
-
}
|
|
1101
|
-
catch {
|
|
1102
|
-
console.log('cannot retrieve mattertagData for tag', mattertagID);
|
|
1103
|
-
}
|
|
1104
|
-
return null;
|
|
1105
|
-
}
|
|
1106
|
-
//
|
|
1107
|
-
// ---------- Viewer related (switch views, go to) ----------
|
|
1108
|
-
//
|
|
1109
|
-
async action_toolbox_floorplan() {
|
|
1110
|
-
if (this.inTransitionMode || this.inTransitionSweep) {
|
|
1111
|
-
console.log('viewer is in transition, cannot go floorplan');
|
|
1112
|
-
return;
|
|
1113
|
-
}
|
|
1114
|
-
try {
|
|
1115
|
-
await this.sdk.Mode.moveTo('mode.floorplan');
|
|
1116
|
-
}
|
|
1117
|
-
catch (e) {
|
|
1118
|
-
console.log('cannot go to floorplan', e);
|
|
1119
|
-
}
|
|
1120
|
-
}
|
|
1121
|
-
action_toolbox_inside_view() {
|
|
1122
|
-
this.sdk.Mode.moveTo('mode.inside');
|
|
1123
|
-
}
|
|
1124
|
-
actionShowAllFloors() {
|
|
1125
|
-
try {
|
|
1126
|
-
this.sdk.Floor.showAll();
|
|
1127
|
-
}
|
|
1128
|
-
catch (e) {
|
|
1129
|
-
console.log('cannot show all floors', e);
|
|
1130
|
-
}
|
|
1131
|
-
}
|
|
1132
|
-
async action_toolbox_dollhouse() {
|
|
1133
|
-
setTimeout(async () => {
|
|
1134
|
-
// console.log("mode: ", this.inTransitionMode, " sweep: ", this.inTransitionSweep);
|
|
1135
|
-
if (this.inTransitionMode || this.inTransitionSweep) {
|
|
1136
|
-
console.log('viewer is in transition, cannot go dollhouse');
|
|
1137
|
-
return;
|
|
1138
|
-
}
|
|
1139
|
-
try {
|
|
1140
|
-
await this.sdk.Mode.moveTo('mode.dollhouse');
|
|
1141
|
-
}
|
|
1142
|
-
catch (e) {
|
|
1143
|
-
console.log('cannot go to dollhouse', e);
|
|
1144
|
-
}
|
|
1145
|
-
}, 1200);
|
|
1146
|
-
}
|
|
1147
|
-
action_toolbox_mesure() {
|
|
1148
|
-
const newState = !this.isMeasureModeOn;
|
|
1149
|
-
this.sdk.Measurements.toggleMode(newState).then(() => {
|
|
1150
|
-
console.log(`Measurement mode is now ${newState ? 'enabled' : 'disabled'}`);
|
|
1151
|
-
});
|
|
1152
|
-
}
|
|
1153
|
-
action_toolbox_cancel_mesure() {
|
|
1154
|
-
if (this.isMeasureModeOn) {
|
|
1155
|
-
this.action_toolbox_mesure();
|
|
1156
|
-
}
|
|
1157
|
-
}
|
|
1158
|
-
async action_go_to_floor(floorName, matterportFloorSequence = null) {
|
|
1159
|
-
if (!this.floors) {
|
|
1160
|
-
console.log('Floor are not loaded yet');
|
|
1161
|
-
return;
|
|
1162
|
-
}
|
|
1163
|
-
// console.log(this.floors);
|
|
1164
|
-
// look up for sequence number (the safest method)
|
|
1165
|
-
let floorMatterport = this.floors.find((floor) => floor.sequence === matterportFloorSequence);
|
|
1166
|
-
if (!floorMatterport) {
|
|
1167
|
-
floorMatterport = this.floors.find((floor) => floorName.includes(floor.name) && floor.name != '');
|
|
1168
|
-
}
|
|
1169
|
-
if (!floorMatterport) {
|
|
1170
|
-
floorMatterport = this.floors.find((floor) => floorName.includes(floor.id));
|
|
1171
|
-
}
|
|
1172
|
-
// console.log(floorMatterport)
|
|
1173
|
-
if (floorMatterport) {
|
|
1174
|
-
let retry = true;
|
|
1175
|
-
while (retry) {
|
|
1176
|
-
try {
|
|
1177
|
-
const floorIndex = await this.sdk.Floor.moveTo(floorMatterport.sequence);
|
|
1178
|
-
// console.log("moved to floorIndex", floorIndex);
|
|
1179
|
-
retry = false;
|
|
1180
|
-
}
|
|
1181
|
-
catch (error) {
|
|
1182
|
-
console.log('Cannot move to Floor', error);
|
|
1183
|
-
await wait(100);
|
|
1184
|
-
}
|
|
1185
|
-
}
|
|
1186
|
-
}
|
|
1187
|
-
else {
|
|
1188
|
-
console.warn('No matterport floor found to move to');
|
|
1189
|
-
}
|
|
1190
|
-
}
|
|
1191
|
-
async action_go_to_sweep(sweep, rotation = null) {
|
|
1192
|
-
if (this.forbiddenSweeps.includes(sweep)) {
|
|
1193
|
-
console.log('user is not allowed to go to this sweep');
|
|
1194
|
-
return;
|
|
1195
|
-
}
|
|
1196
|
-
// console.log("going to sweep", sweep, "with rotation:", rotation);
|
|
1197
|
-
setTimeout(async () => {
|
|
1198
|
-
if (this.inTransitionMode || this.inTransitionSweep) {
|
|
1199
|
-
console.log('Cannot go to sweep, in transition');
|
|
1200
|
-
return;
|
|
1201
|
-
}
|
|
1202
|
-
try {
|
|
1203
|
-
this.inTransitionSweep = true;
|
|
1204
|
-
await this.sdk.Sweep.moveTo(sweep, {
|
|
1205
|
-
transition: 'transition.instant',
|
|
1206
|
-
transitionTime: 1500,
|
|
1207
|
-
});
|
|
1208
|
-
}
|
|
1209
|
-
catch (error) {
|
|
1210
|
-
this.inTransitionSweep = false;
|
|
1211
|
-
console.log('Cannot move to sweep', error);
|
|
1212
|
-
}
|
|
1213
|
-
if (rotation) {
|
|
1214
|
-
await this.sdk.Camera.setRotation(rotation, { speed: 100 }); // speed is degrees per second
|
|
1215
|
-
}
|
|
1216
|
-
}, 1000);
|
|
1217
|
-
}
|
|
1218
|
-
getCurrentSweep() {
|
|
1219
|
-
if (this.poseCamera) {
|
|
1220
|
-
return this.poseCamera.sweep;
|
|
1221
|
-
}
|
|
1222
|
-
return null;
|
|
1223
|
-
}
|
|
1224
|
-
getCurrentCameraPosition() {
|
|
1225
|
-
if (this.poseCamera) {
|
|
1226
|
-
return this.poseCamera;
|
|
1227
|
-
}
|
|
1228
|
-
return null;
|
|
1229
|
-
}
|
|
1230
|
-
setInteractionMode(interactionMode) {
|
|
1231
|
-
this.interactionMode = interactionMode;
|
|
1232
|
-
}
|
|
1233
|
-
getInteractionMode() {
|
|
1234
|
-
return this.interactionMode;
|
|
1235
|
-
}
|
|
1236
|
-
/**
|
|
1237
|
-
* Clear all variables, deletes all tags (when viewer is remover or model changed=reload needed)
|
|
1238
|
-
*/
|
|
1239
|
-
async clearAll() {
|
|
1240
|
-
console.log('removing viewer');
|
|
1241
|
-
this.action_delete_all_mattertags();
|
|
1242
|
-
this.floors = null;
|
|
1243
|
-
this.sweeps = null;
|
|
1244
|
-
this.sdk = null;
|
|
1245
|
-
clearInterval(this.timerPointer);
|
|
1246
|
-
this.forbiddenSweeps = [];
|
|
1247
|
-
this.tagMessengerOn = false;
|
|
1248
|
-
// cancel subscriptions
|
|
1249
|
-
this.pointerButton.removeEventListener('click', this.pointerLeftClickHandler);
|
|
1250
|
-
this.pointerButton.removeEventListener('contextmenu', this.pointerRightClickHandler);
|
|
1251
|
-
// TODO: only For Admins
|
|
1252
|
-
if (this.getCursorPositionButton) {
|
|
1253
|
-
this.getCursorPositionButton.removeEventListener('auxclick', this.pointerMiddleClickHandler);
|
|
1254
|
-
}
|
|
1255
|
-
// TODO: only for dev!
|
|
1256
|
-
if (!!this.getCursorPositionButton &&
|
|
1257
|
-
(document.location.href.indexOf('dev') !== -1 || document.location.href.indexOf('localhost') !== -1)) {
|
|
1258
|
-
clearInterval(this.intervalCursorPointerPosition);
|
|
1259
|
-
}
|
|
1260
|
-
}
|
|
1261
|
-
async removeForbiddenSweeps(forbiddenSweeps) {
|
|
1262
|
-
this.forbiddenSweeps = [...forbiddenSweeps];
|
|
1263
|
-
let removed = 0;
|
|
1264
|
-
await Promise.all(forbiddenSweeps.map(async (sweep) => {
|
|
1265
|
-
try {
|
|
1266
|
-
await this.sdk.Sweep.disable(sweep);
|
|
1267
|
-
removed += 1;
|
|
1268
|
-
}
|
|
1269
|
-
catch (error) {
|
|
1270
|
-
console.log(error);
|
|
1271
|
-
}
|
|
1272
|
-
}));
|
|
1273
|
-
console.log('removed sweeps:', removed);
|
|
1274
|
-
}
|
|
1275
|
-
//
|
|
1276
|
-
// ---------- 3D objects (SDK Bundle only) ----------
|
|
1277
|
-
//
|
|
1278
|
-
async init3DObjectViewer() {
|
|
1279
|
-
return new Promise(async (resolve) => {
|
|
1280
|
-
var [sceneObject] = await this.sdk.Scene.createObjects(1);
|
|
1281
|
-
var node = sceneObject.addNode();
|
|
1282
|
-
// TODO change this 🤮
|
|
1283
|
-
node.addComponent('mp.lights');
|
|
1284
|
-
//node.addComponent('mp.lights');
|
|
1285
|
-
/*node.addComponent('mp.lights');
|
|
1286
|
-
node.addComponent('mp.lights');*/
|
|
1287
|
-
node.start();
|
|
1288
|
-
const nodeControl = sceneObject.addNode();
|
|
1289
|
-
this.objectControl = nodeControl.addComponent('mp.transformControls');
|
|
1290
|
-
nodeControl.start();
|
|
1291
|
-
//this.add3DObject({},null);
|
|
1292
|
-
resolve();
|
|
1293
|
-
});
|
|
1294
|
-
}
|
|
1295
|
-
async add3DObject(obj, mode) {
|
|
1296
|
-
return new Promise(async (resolve) => {
|
|
1297
|
-
const [sceneObject] = await this.sdk.Scene.createObjects(1);
|
|
1298
|
-
// TODO: improvment, regroup all of these in Dynamical Objects Lib
|
|
1299
|
-
// SecurityCamera
|
|
1300
|
-
// NestThermostat
|
|
1301
|
-
// Video
|
|
1302
|
-
let isAnimatedSecurityCamera = false;
|
|
1303
|
-
let isNestThermostat = false;
|
|
1304
|
-
let isSmarterplanPromotionalVideo = false;
|
|
1305
|
-
let isAzimuthalCrown = false;
|
|
1306
|
-
/**
|
|
1307
|
-
* TODO: refacto with an enum or switch/case
|
|
1308
|
-
*/
|
|
1309
|
-
if (obj.object === "security_camera") {
|
|
1310
|
-
isAnimatedSecurityCamera = true;
|
|
1311
|
-
}
|
|
1312
|
-
if (obj.object === "nest_thermostat") {
|
|
1313
|
-
isNestThermostat = true;
|
|
1314
|
-
}
|
|
1315
|
-
if (obj.object === 'video') {
|
|
1316
|
-
isSmarterplanPromotionalVideo = true;
|
|
1317
|
-
}
|
|
1318
|
-
if (obj.object === 'azimuth') {
|
|
1319
|
-
isAzimuthalCrown = true;
|
|
1320
|
-
}
|
|
1321
|
-
const modelNode = sceneObject.addNode();
|
|
1322
|
-
let component = null;
|
|
1323
|
-
const initial = {
|
|
1324
|
-
url: `/assets/3Dobjects/objects/${obj.object}${obj.format.indexOf('.') === -1 ? '.' + obj.format : obj.format}`,
|
|
1325
|
-
// TODO/ store localPosition && localRotation in BDD too (in order to have pertfect initial placement)
|
|
1326
|
-
localRotation: { "x": 0, "y": 0, "z": 0 },
|
|
1327
|
-
// TODO/ store localPosition && localRotation in BDD too (in order to have pertfect initial placement)
|
|
1328
|
-
localPosition: { "x": 0, "y": 0, "z": 0 },
|
|
1329
|
-
visible: true,
|
|
1330
|
-
colliderEnabled: true
|
|
1331
|
-
};
|
|
1332
|
-
switch (obj.format) {
|
|
1333
|
-
case '.obj':
|
|
1334
|
-
case 'obj':
|
|
1335
|
-
component = modelNode.addComponent('mp.objLoader', initial);
|
|
1336
|
-
break;
|
|
1337
|
-
case '.fbx':
|
|
1338
|
-
case 'fbx':
|
|
1339
|
-
component = modelNode.addComponent('mp.fbxLoader', initial);
|
|
1340
|
-
break;
|
|
1341
|
-
case '.gltf':
|
|
1342
|
-
case 'gltf':
|
|
1343
|
-
component = modelNode.addComponent('mp.gltfLoader', initial);
|
|
1344
|
-
break;
|
|
1345
|
-
case '.glb':
|
|
1346
|
-
case 'glb':
|
|
1347
|
-
component = modelNode.addComponent('mp.gltfLoader', initial);
|
|
1348
|
-
break;
|
|
1349
|
-
case '.dae':
|
|
1350
|
-
case 'dae':
|
|
1351
|
-
component = modelNode.addComponent('mp.daeLoader', initial);
|
|
1352
|
-
break;
|
|
1353
|
-
default:
|
|
1354
|
-
console.log('Format not supported');
|
|
1355
|
-
break;
|
|
1356
|
-
}
|
|
1357
|
-
//cache system (i'll try to make it work later ...)
|
|
1358
|
-
/*let objContentsJSON = JSON.stringify(modelNode);
|
|
1359
|
-
console.log(modelNode);
|
|
1360
|
-
console.log(objContentsJSON);
|
|
1361
|
-
console.log(JSON.stringify(modelNode));
|
|
1362
|
-
localStorage.setItem(`testObj_${obj.object}`, objContentsJSON);
|
|
1363
|
-
console.log("stored ?!");
|
|
1364
|
-
console.log(typeof modelNode);*/
|
|
1365
|
-
//let dataS = serialijse.serialize(modelNode);
|
|
1366
|
-
//console.log(dataS);
|
|
1367
|
-
// Use 3 ?! Ambient lightings
|
|
1368
|
-
if (this.noLightForObjects) {
|
|
1369
|
-
const lightsNode = sceneObject.addNode();
|
|
1370
|
-
lightsNode.addComponent('mp.ambientLight', {
|
|
1371
|
-
intensity: 1,
|
|
1372
|
-
color: { r: 1.0, g: 1.0, b: 1.0 },
|
|
1373
|
-
});
|
|
1374
|
-
this.noLightForObjects = false;
|
|
1375
|
-
}
|
|
1376
|
-
modelNode.obj3D.position.set(obj.position.x, obj.position.y, obj.position.z);
|
|
1377
|
-
modelNode.obj3D.rotation.set(obj.rotation.x, obj.rotation.y, obj.rotation.z);
|
|
1378
|
-
modelNode.obj3D.scale.set(obj.scale.x, obj.scale.y, obj.scale.z);
|
|
1379
|
-
if (isAzimuthalCrown) {
|
|
1380
|
-
/*if(modelNode.obj3D.scale.x >= 1.0){
|
|
1381
|
-
modelNode.obj3D.scale.set(obj.scale.x/5000, obj.scale.y/5000, obj.scale.z/5000);
|
|
1382
|
-
}*/
|
|
1383
|
-
this.azimuthalCrown = modelNode;
|
|
1384
|
-
this.displayAzimutalCrown();
|
|
1385
|
-
}
|
|
1386
|
-
// By defaut, during creation, object has translation gizmo
|
|
1387
|
-
// => User has to click on lateral panel, and save its position after playing with it!
|
|
1388
|
-
if (mode && !isNestThermostat) {
|
|
1389
|
-
this.attachGizmoControlTo3DObject(modelNode, sceneObject, mode, true, true).catch((e) => console.log(e));
|
|
1390
|
-
}
|
|
1391
|
-
if (this.lastObject3D && typeof this.lastObject3D.id === 'string') {
|
|
1392
|
-
// prompt ThreeJS UUID to match our last generated object if new (not from Db)
|
|
1393
|
-
modelNode.obj3D.uuid = this.lastObject3D.id;
|
|
1394
|
-
}
|
|
1395
|
-
else if (obj.id) {
|
|
1396
|
-
modelNode.obj3D.uuid = obj.id;
|
|
1397
|
-
}
|
|
1398
|
-
this.lastObject3D = modelNode.obj3D;
|
|
1399
|
-
// Store this in memory Map dictionnary
|
|
1400
|
-
//console.log("Adding object: ");
|
|
1401
|
-
//console.log(modelNode);
|
|
1402
|
-
this.dictionnaryObjects3D.set(modelNode.obj3D.uuid, modelNode);
|
|
1403
|
-
this.dictionnarySceneObjects3D.set(modelNode.obj3D.uuid, sceneObject);
|
|
1404
|
-
/*this.sdk.Camera.pose.subscribe(
|
|
1405
|
-
function (pose: any) {
|
|
1406
|
-
//console.log("in callback")
|
|
1407
|
-
//console.log(this.lastCameraPosition)
|
|
1408
|
-
//console.log(pose.position)
|
|
1409
|
-
if((pose.position.x == this.lastCameraPosition.x && pose.position.y == this.lastCameraPosition.y && pose.position.z == this.lastCameraPosition.z) || this.inTransitionMode || this.inTransitionSweep){
|
|
1410
|
-
//console.log("returning")
|
|
1411
|
-
return;
|
|
1412
|
-
}
|
|
1413
|
-
console.log("camera pos:",pose.position);
|
|
1414
|
-
this.lastCameraPosition = {...pose.position};
|
|
1415
|
-
if(this.lastObject3D){
|
|
1416
|
-
this.lastObject3D.position.set(pose.position.x+.5,pose.position.y+.5,pose.position.z+.5);
|
|
1417
|
-
}
|
|
1418
|
-
}.bind(this));*/
|
|
1419
|
-
if (isAnimatedSecurityCamera) {
|
|
1420
|
-
const animator = modelNode.addComponent('mp.securityCamera', {
|
|
1421
|
-
"nearPlane": 0.1,
|
|
1422
|
-
"farPlane": 10,
|
|
1423
|
-
"horizontalFOV": 52,
|
|
1424
|
-
"aspect": 1.7777777777777777,
|
|
1425
|
-
"localPosition": {
|
|
1426
|
-
"x": 0.3,
|
|
1427
|
-
"y": 0.18,
|
|
1428
|
-
"z": 0
|
|
1429
|
-
},
|
|
1430
|
-
"localRotation": {
|
|
1431
|
-
"x": -15,
|
|
1432
|
-
"y": -90,
|
|
1433
|
-
"z": 0
|
|
1434
|
-
},
|
|
1435
|
-
"color": 65280,
|
|
1436
|
-
"panPeriod": 5,
|
|
1437
|
-
"panAngle": -45
|
|
1438
|
-
});
|
|
1439
|
-
const modelOutput = sceneObject.addPath({
|
|
1440
|
-
id: 'animated-model',
|
|
1441
|
-
type: this.sdk.Scene.PathType.OUTPUT,
|
|
1442
|
-
node: modelNode,
|
|
1443
|
-
component: animator,
|
|
1444
|
-
property: 'objectRoot'
|
|
1445
|
-
});
|
|
1446
|
-
this.securityCameraAnimator = animator;
|
|
1447
|
-
if (!obj.viewFrustum) {
|
|
1448
|
-
setTimeout(() => {
|
|
1449
|
-
animator.toggleViewFrustum();
|
|
1450
|
-
}, 1000);
|
|
1451
|
-
}
|
|
1452
|
-
}
|
|
1453
|
-
if (isNestThermostat) {
|
|
1454
|
-
// TODO: use bindPath instead using MP sdk classes (see Security Camera example)
|
|
1455
|
-
// for TV uses CanvasImage below!
|
|
1456
|
-
// const ci = new CanvasImage();
|
|
1457
|
-
// ci.onInit();
|
|
1458
|
-
const cv = new CanvasRenderer();
|
|
1459
|
-
cv.onInit();
|
|
1460
|
-
const plane = new PlaneRenderer();
|
|
1461
|
-
plane.outputs = {
|
|
1462
|
-
objectRoot: new Object3D(),
|
|
1463
|
-
collider: new Object3D()
|
|
1464
|
-
};
|
|
1465
|
-
const inputTexture = cv.outputs.texture;
|
|
1466
|
-
plane.setRootScene(this.threeJSScene);
|
|
1467
|
-
plane.onInit(modelNode, inputTexture);
|
|
1468
|
-
const sc = new NestThermostat();
|
|
1469
|
-
sc.setComponent(component, plane, cv);
|
|
1470
|
-
sc.setRootScene(this.threeJSScene);
|
|
1471
|
-
sc.onInit(modelNode, plane, inputTexture);
|
|
1472
|
-
cv.setCanvasNestThermostatPainter(sc);
|
|
1473
|
-
setTimeout(() => {
|
|
1474
|
-
sc.inputs.loadingState = "Loaded";
|
|
1475
|
-
sc.onInputsUpdated();
|
|
1476
|
-
}, 5000);
|
|
1477
|
-
}
|
|
1478
|
-
if (isSmarterplanPromotionalVideo) {
|
|
1479
|
-
// TODO: use bindPath instead using MP sdk classes (see Security Camera example)
|
|
1480
|
-
const sc = new TvPlayer();
|
|
1481
|
-
sc.setComponent(component);
|
|
1482
|
-
sc.onInit(modelNode);
|
|
1483
|
-
setTimeout(() => {
|
|
1484
|
-
sc.inputs.loadingState = "Loaded";
|
|
1485
|
-
sc.onInputsUpdated();
|
|
1486
|
-
}, 5000);
|
|
1487
|
-
}
|
|
1488
|
-
sceneObject.start();
|
|
1489
|
-
resolve(this.lastObject3D);
|
|
1490
|
-
});
|
|
1491
|
-
}
|
|
1492
|
-
toggleObjectVisibility(objectId) {
|
|
1493
|
-
let obj = this.dictionnaryObjects3D.get(objectId);
|
|
1494
|
-
obj.obj3D.visible = !obj.obj3D.visible;
|
|
1495
|
-
}
|
|
1496
|
-
isObjectVisible(objectId) {
|
|
1497
|
-
let obj = this.dictionnaryObjects3D.get(objectId);
|
|
1498
|
-
return obj.obj3D.visible;
|
|
1499
|
-
}
|
|
1500
|
-
async pointCameraTo3DObject(objectId) {
|
|
1501
|
-
let obj = this.dictionnaryObjects3D.get(objectId);
|
|
1502
|
-
//We create a temporary Tag
|
|
1503
|
-
const poiObject = {
|
|
1504
|
-
coordinate: JSON.stringify(obj.obj3D.position),
|
|
1505
|
-
type: PoiType.OBJECT3D,
|
|
1506
|
-
elementID: objectId, //todo: be careful with this
|
|
1507
|
-
};
|
|
1508
|
-
const objectDb = { id: objectId };
|
|
1509
|
-
try {
|
|
1510
|
-
await this.createMattertagFromPOI(PoiType.OBJECT3D, objectDb, poiObject);
|
|
1511
|
-
}
|
|
1512
|
-
catch (err) { }
|
|
1513
|
-
//Not really necessary anymore since the tag will disappear quick
|
|
1514
|
-
//this.sdk.Tag.editOpacity(mattertagID, 0.0);//opacity);
|
|
1515
|
-
//this.sdk.Tag.allowAction(mattertagID, {}); //disables every action
|
|
1516
|
-
let result = this.getTagFromElementId(objectId);
|
|
1517
|
-
await this.goToTag(result.tag);
|
|
1518
|
-
this.deleteLastMattertag();
|
|
1519
|
-
}
|
|
1520
|
-
getSceneNodeFromObject3DId(uuid) {
|
|
1521
|
-
return this.dictionnarySceneObjects3D.get(uuid);
|
|
1522
|
-
}
|
|
1523
|
-
async displayAzimutalCrown() {
|
|
1524
|
-
if (this.azimuthalCrown) {
|
|
1525
|
-
this.azimuthalCrown.obj3D.position.set(this.poseCamera.position.x, this.poseCamera.position.y, this.poseCamera.position.z);
|
|
1526
|
-
}
|
|
1527
|
-
}
|
|
1528
|
-
async attachGizmoControlTo3DObject(modelNode, sceneObject, mode, visible, isNewObject) {
|
|
1529
|
-
// Create a scene node with a transform control component.
|
|
1530
|
-
let node = null;
|
|
1531
|
-
node = sceneObject.addNode();
|
|
1532
|
-
if (!node) {
|
|
1533
|
-
const [sceneObject] = await this.sdk.Scene.createObjects(1);
|
|
1534
|
-
node = sceneObject.addNode();
|
|
1535
|
-
}
|
|
1536
|
-
const myControl = node.addComponent('mp.transformControls');
|
|
1537
|
-
node.start();
|
|
1538
|
-
//
|
|
1539
|
-
// // Make the transform control visible so that the user can manipulate the control selection.
|
|
1540
|
-
myControl.transformControls.visible = visible;
|
|
1541
|
-
//
|
|
1542
|
-
// // Attach the model to the transform control
|
|
1543
|
-
myControl.inputs.selection = modelNode;
|
|
1544
|
-
//
|
|
1545
|
-
// // set 'translate' mode to position the selection.
|
|
1546
|
-
myControl.inputs.mode = mode;
|
|
1547
|
-
modelNode.obj3D.controls = myControl; // store gizmoCtrl inside object
|
|
1548
|
-
if (isNewObject) { //i keep the current solution for new objects
|
|
1549
|
-
if (!this.lastObject3D || !this.lastObject3D.controls) {
|
|
1550
|
-
try {
|
|
1551
|
-
modelNode.obj3D.uuid = this.lastObject3D.uuid || this.lastObject3D.id;
|
|
1552
|
-
}
|
|
1553
|
-
catch (e) {
|
|
1554
|
-
console.log(`id obj in Scene was not assigned to id from DB since`);
|
|
1555
|
-
}
|
|
1556
|
-
this.lastObject3D = modelNode.obj3D;
|
|
1557
|
-
}
|
|
1558
|
-
}
|
|
1559
|
-
else { //objects already in place have to become the "lastObject" (i think?)
|
|
1560
|
-
console.log("in my solution !");
|
|
1561
|
-
console.log(modelNode);
|
|
1562
|
-
console.log(modelNode.obj3D.uuid);
|
|
1563
|
-
this.lastObject3D = modelNode.obj3D;
|
|
1564
|
-
}
|
|
1565
|
-
return modelNode;
|
|
1566
|
-
}
|
|
1567
|
-
removeGizmoFromLastObject() {
|
|
1568
|
-
this.lastObject3D.controls.transformControls.visible = false;
|
|
1569
|
-
this.lastObject3D.controls.transformControls.dispose();
|
|
1570
|
-
this.lastObject3D.controls = null;
|
|
1571
|
-
}
|
|
1572
|
-
toggleViewFrustum() {
|
|
1573
|
-
this.securityCameraAnimator.toggleViewFrustum();
|
|
1574
|
-
}
|
|
1575
|
-
}
|
|
1576
|
-
MatterportService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.2.7", ngImport: i0, type: MatterportService, deps: [{ token: 'config' }, { token: i1.Router }, { token: i1.ActivatedRoute }, { token: i2.BaseVisibilityService }, { token: i0.NgZone }], target: i0.ɵɵFactoryTarget.Injectable });
|
|
1577
|
-
MatterportService.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "13.2.7", ngImport: i0, type: MatterportService, providedIn: 'root' });
|
|
1578
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.2.7", ngImport: i0, type: MatterportService, decorators: [{
|
|
1579
|
-
type: Injectable,
|
|
1580
|
-
args: [{
|
|
1581
|
-
providedIn: 'root',
|
|
1582
|
-
}]
|
|
1583
|
-
}], ctorParameters: function () { return [{ type: i3.Config, decorators: [{
|
|
1584
|
-
type: Inject,
|
|
1585
|
-
args: ['config']
|
|
1586
|
-
}] }, { type: i1.Router }, { type: i1.ActivatedRoute }, { type: i2.BaseVisibilityService }, { type: i0.NgZone }]; } });
|
|
1587
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"matterport.service.js","sourceRoot":"","sources":["../../../../../projects/ngx-smarterplan-core/src/lib/services/matterport.service.ts"],"names":[],"mappings":"AAAA,OAAO,EAAC,MAAM,EAAE,UAAU,EAAS,MAAM,eAAe,CAAC;AAEzD,OAAO,EAAC,OAAO,EAAC,MAAM,MAAM,CAAC;AAC7B,2CAA2C;AAC3C,OAAO,EACL,2BAA2B,EAC3B,eAAe,EACf,IAAI,GACL,MAAM,oBAAoB,CAAC;AAC5B,OAAO,EAAC,aAAa,EAAC,MAAM,kBAAkB,CAAC;AAC/C,OAAO,EAEL,WAAW,EAEX,mBAAmB,EAEnB,OAAO,EACP,QAAQ,EACR,SAAS,EACT,kBAAkB,GACnB,MAAM,kBAAkB,CAAC;AAC1B,OAAO,EAAC,QAAQ,EAAC,MAAM,OAAO,CAAC;AAI/B,OAAO,EAAC,UAAU,EAAC,MAAM,kBAAkB,CAAC;AAG5C,OAAO,EAAC,cAAc,EAAC,MAAM,yDAAyD,CAAC;AACvF,OAAO,EAAC,aAAa,EAAC,MAAM,wDAAwD,CAAC;AACrF,OAAO,EAAC,cAAc,EAAC,MAAM,yDAAyD,CAAC;AAGvF,OAAO,EAAC,QAAQ,EAAC,MAAM,6CAA6C,CAAC;;;;;AAwBrE,MAAM,OAAO,iBAAiB;IAoJ5B,YACoB,MAAc,EACxB,MAAc,EACd,WAA2B,EAC3B,iBAAwC,EACxC,MAAc;QAHd,WAAM,GAAN,MAAM,CAAQ;QACd,gBAAW,GAAX,WAAW,CAAgB;QAC3B,sBAAiB,GAAjB,iBAAiB,CAAuB;QACxC,WAAM,GAAN,MAAM,CAAQ;QAvJhB,UAAK,GAAU,EAAE,CAAC,CAAA,kBAAkB;QAEpC,UAAK,GAAiB,EAAE,CAAC;QAU1B,uBAAkB,GAAQ,EAAC,CAAC,EAAC,GAAG,EAAC,CAAC,EAAC,GAAG,EAAC,CAAC,EAAC,GAAG,EAAC,CAAC;QAQ7C,kCAA6B,GAAG,KAAK,CAAC;QAK9C,eAAe;QACP,oBAAe,GAAG,KAAK,CAAC;QAExB,oBAAe,GAAW,kBAAkB,CAAC,OAAO,CAAC;QAE7D,8DAA8D;QACtD,iBAAY,GAAkB,EAAE,CAAC;QAEzC,0DAA0D;QAClD,oBAAe,GAA+B,IAAI,GAAG,EAAE,CAAC;QAExD,yBAAoB,GAAqB,IAAI,GAAG,EAAE,CAAC;QAEnD,8BAAyB,GAAqB,IAAI,GAAG,EAAE,CAAC;QAIxD,gBAAW,GAAG,EAAE,CAAC;QAEjB,yBAAoB,GAAa,EAAE,CAAC;QAEpC,eAAU,GAAG;YACnB,KAAK,EAAE,GAAG;YACV,MAAM,EAAE,GAAG;SACZ,CAAC;QAEM,eAAU,GAAG;YACnB,UAAU,EAAE,KAAK;YACjB,MAAM,EAAE,IAAI;SACb,CAAC;QAEK,oBAAe,GAAW,EAAE,CAAC;QAM7B,iBAAY,GAAoB,IAAI,OAAO,EAAE,CAAC;QAErD,+BAA+B;QACxB,sBAAiB,GAAiB,IAAI,OAAO,EAAE,CAAC;QAoBvD,oBAAe,GAAa,EAAE,CAAC;QAExB,cAAS,GAA8C,IAAI,OAAO,EAAE,CAAC;QAQrE,qBAAgB,GAAG,KAAK,CAAC;QAEzB,sBAAiB,GAAG,KAAK,CAAC;QAEzB,sBAAiB,GAAG,IAAI,CAAC;QAE1B,sBAAiB,GAAe,UAAU,CAAC,OAAO,CAAC;QAEnD,wBAAmB,GAAG,IAAI,OAAO,EAAc,CAAC;QAEhD,cAAS,GAAG,IAAI,OAAO,EAAU,CAAC;QAEzC,mBAAc,GAAG,KAAK,CAAC;QAQvB;;WAEG;QACH,4BAAuB,GAAG,GAAG,EAAE;YAC7B,IAAI,IAAI,CAAC,iBAAiB,EAAE;gBAC1B,MAAM,aAAa,GAAG,IAAI,CAAC,eAAe,CAAC,GAAG,CAAC,IAAI,CAAC,iBAAiB,CAAC,CAAC;gBACvE,aAAa,CAAC,WAAW,CAAC,EAAC,GAAG,IAAI,CAAC,cAAc,CAAC,QAAQ,EAAC,CAAC,CAAC,CAAC,2BAA2B;gBACzF,aAAa,CAAC,SAAS,CAAC,EAAC,GAAG,IAAI,CAAC,cAAc,CAAC,MAAM,EAAC,CAAC,CAAC,CAAC,2BAA2B;gBACrF,IAAI,CAAC,eAAe,CAAC,GAAG,CAAC,IAAI,CAAC,iBAAiB,EAAE,aAAa,CAAC,CAAC;gBAChE,IAAI,CAAC,8BAA8B,CAAC,IAAI,CAAC,iBAAiB,CAAC,CAAC;aAC7D;YACD,IAAI,CAAC,oBAAoB,EAAE,CAAC;QAC9B,CAAC,CAAC;QAEF,6BAAwB,GAAG,CAAC,CAAC,EAAE,EAAE;YAC/B,CAAC,CAAC,cAAc,EAAE,CAAC;YACnB,IAAI,CAAC,qBAAqB,EAAE,CAAC;YAC7B,KAAK,CAAC,kBAAkB,CAAC,CAAC;QAC5B,CAAC,CAAC;QAEF,8BAAyB,GAAG,CAAC,CAAC,EAAE,EAAE;YAChC,IAAI,CAAC,8BAA8B,CAAC,SAAS,GAAG;SAC3C,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,cAAc,CAAC,QAAQ,CAAC;;SAEhD,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,cAAc,CAAC,MAAM,CAAC;;SAE9C,IAAI,CAAC,cAAc,CAAC,OAAO,EAAE,CAAC;YACnC,iEAAiE;YACjE,IAAI,CAAC,uBAAuB,CAAC,KAAK,CAAC,OAAO,GAAG,MAAM,CAAC;QACtD,CAAC,CAAA;QASC,IAAI,CAAC,MAAM,GAAG,MAAM,CAAC;QAErB,sBAAsB;QACtB,IAAI,CAAC,CAAC,IAAI,CAAC,uBAAuB;YAChC,CAAC,QAAQ,CAAC,QAAQ,CAAC,IAAI,CAAC,OAAO,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC,IAAI,QAAQ,CAAC,QAAQ,CAAC,IAAI,CAAC,OAAO,CAAC,WAAW,CAAC,KAAK,CAAC,CAAC,CAAC,EAAE;YACtG,IAAI,CAAC,6BAA6B,GAAG,WAAW,CAAC,GAAG,EAAE;gBACpD,IAAI,CAAC,IAAI,CAAC,cAAc,EAAE;oBACxB,OAAO;iBACR;gBAED,MAAM,QAAQ,GAAG,IAAI,CAAC,cAAc,CAAC,IAAI,GAAG,IAAI,CAAC;gBACjD,IAAI,IAAI,IAAI,EAAE,CAAC,OAAO,EAAE,GAAG,QAAQ,EAAE;oBACnC,IAAI,IAAI,CAAC,6BAA6B,EAAE;wBACtC,OAAO;qBACR;oBAED,MAAM,IAAI,GAAG;wBACX,CAAC,EAAE,IAAI,CAAC,SAAS,CAAC,WAAW;wBAC7B,CAAC,EAAE,IAAI,CAAC,SAAS,CAAC,YAAY;qBAC/B,CAAC;oBACF,MAAM,KAAK,GAAG,IAAI,CAAC,GAAG,CAAC,UAAU,CAAC,aAAa,CAAC,IAAI,CAAC,cAAc,CAAC,QAAQ,EAAE,IAAI,CAAC,UAAU,EAAE,IAAI,CAAC,CAAC;oBACrG,IAAI,CAAC,uBAAuB,CAAC,KAAK,CAAC,IAAI,GAAG,GAAG,KAAK,CAAC,CAAC,GAAG,EAAE,IAAI,CAAC;oBAC9D,IAAI,CAAC,uBAAuB,CAAC,KAAK,CAAC,GAAG,GAAG,GAAG,KAAK,CAAC,CAAC,GAAG,EAAE,IAAI,CAAC;oBAC7D,IAAI,CAAC,uBAAuB,CAAC,KAAK,CAAC,OAAO,GAAG,OAAO,CAAC;oBACrD,8DAA8D;oBAC9D,IAAI,CAAC,6BAA6B,GAAG,IAAI,CAAC;iBAC3C;YACH,CAAC,EAAE,GAAG,CAAC,CAAC;SACT;IACH,CAAC;IA3GD,IAAW,cAAc;QACvB,OAAO,IAAI,CAAC,eAAe,CAAC;IAC9B,CAAC;IAED,IAAW,cAAc,CAAC,KAAa;QACrC,IAAI,CAAC,eAAe,GAAG,KAAK,CAAC;IAC/B,CAAC;IAuGD;;;;;OAKG;IACH,KAAK,CAAC,OAAO,CACX,UAA0B,EAC1B,SAAmB,QAAQ,CAAC,IAAI;QAEhC,IAAI,IAAI,CAAC,GAAG,EAAE;YACZ,kCAAkC;YAClC,MAAM,IAAI,CAAC,4BAA4B,EAAE,CAAC;SAC3C;QACD,IAAI,CAAC,UAAU,GAAG,UAAU,CAAC;QAC7B,IAAI,CAAC,QAAQ,GAAG,MAAM,CAAC;QACvB,OAAO,IAAI,OAAO,CAAC,CAAC,OAAO,EAAE,MAAM,EAAE,EAAE;YACrC,wBAAwB;YACxB,IAAI,CAAC,aAAa,GAAG,QAAQ,CAAC,aAAa,CACzC,uBAAuB,CACT,CAAC;YACjB,IAAI,CAAC,SAAS,GAAG,QAAQ,CAAC,aAAa,CAAC,gBAAgB,CAAgB,CAAC;YACzE,MAAM,MAAM,GAAG,QAAQ,CAAC,aAAa,CACnC,gBAAgB,CACI,CAAC;YACvB,IAAI,CAAC,MAAM,EAAE;gBACX,OAAO;aACR;YAED,gBAAgB;YAChB,IAAI,IAAI,CAAC,aAAa,EAAE;gBACtB,IAAI,CAAC,aAAa,CAAC,gBAAgB,CACjC,OAAO,EACP,IAAI,CAAC,uBAAuB,CAC7B,CAAC;gBACF,wBAAwB;gBACxB,IAAI,CAAC,aAAa,CAAC,gBAAgB,CACjC,aAAa,EACb,IAAI,CAAC,wBAAwB,CAC9B,CAAC;aACH;YACD,wBAAwB;YACxB,IAAI,CAAC,uBAAuB,GAAG,QAAQ,CAAC,aAAa,CACnD,SAAS,CACK,CAAC;YACjB,IAAI,CAAC,8BAA8B,GAAG,QAAQ,CAAC,aAAa,CAC1D,OAAO,CACO,CAAC;YACjB,IAAI,IAAI,CAAC,uBAAuB,EAAE;gBAChC,qDAAqD;gBACrD,IAAI,CAAC,uBAAuB,CAAC,gBAAgB,CAC3C,OAAO,EACP,IAAI,CAAC,yBAAyB,CAC/B,CAAC;aACH;YAED,WAAW;YACX,OAAO,CAAC,GAAG,CAAC,wBAAwB,CAAC,CAAC;YACtC,MAAM,cAAc,GAAG,MAAM,CAAC,aAAa,CAAC;YAC5C,MAAM,CAAC,gBAAgB,CACrB,MAAM,EACN,KAAK;gBACH,IAAI;oBACF,IAAI,CAAC,GAAG,GAAG,MAAM,cAAc,CAAC,MAAM,CAAC,OAAO,CAC5C,cAAc,EACd,2BAA2B,EAC3B,KAAK,CACN,CAAC;iBACH;gBAAC,OAAO,CAAC,EAAE;oBACV,OAAO,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;oBACjB,OAAO;iBACR;gBAED,gDAAgD;gBAChD,QAAQ,MAAM,EAAE;oBACd,KAAK,QAAQ,CAAC,IAAI;wBAChB,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC,eAAe,CAC5B,aAAa,EACb,IAAI,CAAC,MAAM,CAAC,SAAS,CAAC,WAAW,CAClC,CAAC;wBACF,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC,eAAe,CAC5B,gBAAgB,EAChB,IAAI,CAAC,MAAM,CAAC,SAAS,CAAC,cAAc,CACrC,CAAC;wBACF,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC,eAAe,CAC5B,cAAc,EACd,IAAI,CAAC,MAAM,CAAC,SAAS,CAAC,YAAY,CACnC,CAAC;wBACF,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC,eAAe,CAC5B,WAAW,EACX,IAAI,CAAC,MAAM,CAAC,SAAS,CAAC,SAAS,CAChC,CAAC;wBACF,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC,eAAe,CAC5B,eAAe,EACf,IAAI,CAAC,MAAM,CAAC,SAAS,CAAC,aAAa,CACpC,CAAC;wBAEF,IAAI,CAAC,GAAG,CAAC,YAAY,CAAC,IAAI,CAAC,SAAS,CAAC;4BACnC,OAAO,EAAE,UACP,KAAU,EACV,IAAqB,EACrB,UAAe;gCAEf,eAAe;gCACf,sCAAsC;gCACtC,aAAa;gCACb,YAAY;gCACZ,KAAK;gCACL,mCAAmC;gCACnC,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC,MAAM,CAAC;4BACjC,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC;4BACZ,mBAAmB,EAAE,UAAU,UAAe;gCAC5C,+DAA+D;gCAC/D,IAAI,CAAC,6BAA6B,EAAE,CAAC;4BACvC,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC;yBACb,CAAC,CAAC;wBACH,IAAI,CAAC,GAAG,CAAC,YAAY,CAAC,IAAI,CAAC,SAAS,CAClC,UAAU,oBAAqC;4BAC7C,qCAAqC;4BACrC,IAAI,CAAC,eAAe,GAAG,oBAAoB,CAAC,MAAM,CAAC;4BACnD,sFAAsF;wBACxF,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CACb,CAAC;wBACF,MAAM;oBACR,KAAK,QAAQ,CAAC,MAAM;wBAClB,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC,eAAe,CAC5B,WAAW,EACX,IAAI,CAAC,MAAM,CAAC,SAAS,CAAC,SAAS,CAChC,CAAC;wBACF,MAAM;oBACR,KAAK,QAAQ,CAAC,KAAK;wBACjB,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC,eAAe,CAC5B,qBAAqB,EACrB,IAAI,CAAC,MAAM,CAAC,SAAS,CAAC,mBAAmB,CAC1C,CAAC;wBACF,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC,eAAe,CAC5B,uBAAuB,EACvB,IAAI,CAAC,MAAM,CAAC,SAAS,CAAC,qBAAqB,CAC5C,CAAC;wBACF,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC,eAAe,CAC5B,kBAAkB,EAClB,IAAI,CAAC,MAAM,CAAC,SAAS,CAAC,gBAAgB,CACvC,CAAC;wBACF,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC,eAAe,CAC5B,kBAAkB,EAClB,IAAI,CAAC,MAAM,CAAC,SAAS,CAAC,gBAAgB,CACvC,CAAC;wBACF,MAAM;oBACR;wBACE,MAAM;iBACT;gBAED,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC,eAAe,CAC5B,eAAe,EACf,IAAI,CAAC,MAAM,CAAC,SAAS,CAAC,aAAa,CACpC,CAAC;gBAEF,kHAAkH;gBAClH,IAAI,CAAC,GAAG,CAAC,IAAI,CAAC,OAAO,CAAC,SAAS,CAAC,CAAC,WAAW,EAAE,EAAE;oBAC9C,OAAO,CAAC,GAAG,CAAC,sBAAsB,IAAI,CAAC,SAAS,CAAC,WAAW,CAAC,EAAE,CAAC,CAAC;gBACnE,CAAC,CAAC,CAAC;gBAEH,wBAAwB;gBACxB,IAAI,CAAC,GAAG,CAAC,OAAO,CAAC,YAAY,CAAC,SAAS,CACrC,UAAU,YAAiB;oBACzB,IAAI,CAAC,cAAc,GAAG,YAAY,CAAC;oBACnC,IAAI,CAAC,cAAc,CAAC,IAAI,GAAG,IAAI,IAAI,EAAE,CAAC,OAAO,EAAE,CAAC;oBAChD,IAAI,CAAC,IAAI,CAAC,yBAAyB,EAAE;wBACnC,IAAI,CAAC,yBAAyB,GAAG;4BAC/B,CAAC,EAAE,CAAC;4BACJ,CAAC,EAAE,CAAC;4BACJ,CAAC,EAAE,CAAC;yBACL,CAAC;qBACH;oBAED,IACE,IAAI,CAAC,eAAe,KAAK,kBAAkB,CAAC,OAAO;wBACnD,IAAI,CAAC,iBAAiB,EACtB;wBACA,8DAA8D;wBAC9D,2DAA2D;wBAC3D,IAAI,CAAC,oBAAoB,CAAC,IAAI,CAAC,iBAAiB,CAAC,CAAC;qBACnD;oBACD,IAAI,CAAC,CAAC,IAAI,CAAC,uBAAuB,IAAI,CAAC,CAAC,IAAI,CAAC,uBAAuB,CAAC,KAAK;wBAC1E,CAAC,QAAQ,CAAC,GAAG,CAAC,OAAO,CAAC,4BAA4B,CAAC,KAAK,CAAC,CAAC,IAAI,QAAQ,CAAC,QAAQ,CAAC,IAAI,CAAC,OAAO,CAAC,WAAW,CAAC,KAAK,CAAC,CAAC,CAAC,EAAE;wBACjH,IAAI,CAAC,uBAAuB,CAAC,KAAK,CAAC,OAAO,GAAG,MAAM,CAAC;wBACpD,IAAI,CAAC,6BAA6B,GAAG,KAAK,CAAC;qBAC5C;gBACH,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CACb,CAAC;gBAEF,aAAa;gBACb,IAAI,CAAC,GAAG,CAAC,IAAI,CAAC,OAAO,CAAC,SAAS,CAAC,CAAC,IAAI,EAAE,EAAE;oBACvC,IAAI,CAAC,iBAAiB,GAAG,KAAK,CAAC;oBAC/B,QAAQ,IAAI,EAAE;wBACZ,KAAK,gBAAgB;4BACnB,IAAI,CAAC,iBAAiB,GAAG,UAAU,CAAC,SAAS,CAAC;4BAC9C,MAAM;wBACR,KAAK,gBAAgB;4BACnB,IAAI,CAAC,iBAAiB,GAAG,UAAU,CAAC,SAAS,CAAC;4BAC9C,MAAM;wBACR,KAAK,aAAa;4BAChB,IAAI,CAAC,iBAAiB,GAAG,UAAU,CAAC,MAAM,CAAC;4BAC3C,MAAM;wBACR,KAAK,oBAAoB;4BACvB,IAAI,CAAC,iBAAiB,GAAG,UAAU,CAAC,aAAa,CAAC;4BAClD,MAAM;wBACR;4BACE,IAAI,CAAC,iBAAiB,GAAG,UAAU,CAAC,OAAO,CAAC;qBAC/C;oBACD,IAAI,CAAC,mBAAmB,CAAC,IAAI,CAAC,IAAI,CAAC,iBAAiB,CAAC,CAAC;gBACxD,CAAC,CAAC,CAAC;gBAEH,qBAAqB;gBACrB,IAAI,CAAC,GAAG,CAAC,MAAM,CAAC,IAAI,CAAC,SAAS,CAC5B,SAAS,MAAM,CAAC,IAAS;oBACvB,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC;oBAEvB,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;oBAClC,sDAAsD;oBACtD,oDAAoD;oBACpD,4CAA4C;gBAC9C,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CACb,CAAC;gBAEF,sBAAsB;gBACtB,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC,IAAI,CAAC,SAAS,CAAC;oBAC5B,mBAAmB,EAAE,SAAS,MAAM,CAAC,UAAc;wBACjD,2CAA2C;wBAC3C,IAAI,CAAC,MAAM,GAAG,MAAM,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;oBACxC,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC;iBACb,CAAC,CAAC;gBAEH,6BAA6B;gBAC7B,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC,OAAO,CAAC,SAAS,CAC9B,SAAS,MAAM,CAAC,YAA6B;oBAC3C,4CAA4C;oBAC5C,IAAI,YAAY,CAAC,GAAG,KAAK,EAAE,EAAE;wBAC3B,8DAA8D;qBAC/D;yBAAM;wBACL,mDAAmD;wBACnD,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,CAAC;qBAC1C;gBACH,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CACb,CAAC;gBAEF,0BAA0B;gBAC1B,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC,IAAI,CAAC,SAAS,CAAC;oBAC5B,mBAAmB,EAAE,SAAS,GAAG,CAAY,UAAe;wBAC1D,IAAI,CAAC,MAAM,GAAG,MAAM,CAAC,MAAM,CAAC,UAAU,CAAC,CAAC;oBAC1C,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC;iBACb,CAAC,CAAC;gBAEH,iCAAiC;gBACjC,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,WAAW,EAAE,CAAC,GAAW,EAAE,EAAE;oBACvC,yBAAyB;oBACzB,IAAI;wBACF,MAAM,aAAa,GAAG,IAAI,CAAC,eAAe,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;wBACpD,IAAI,CAAC,aAAa,EAAC;4BACjB,OAAO,CAAC,GAAG,CAAC,6BAA6B,CAAC,CAAC;4BAC3C,OAAO;yBACR;wBACD,MAAM,GAAG,GAAG,UAAU,CAAC,mBAAmB,CACxC,aAAa,CAAC,SAAS,EAAE,EACzB,aAAa,CAAC,OAAO,EAAE,CACxB,CAAC;wBACF,IAAI,GAAG,KAAK,EAAE,EAAE;4BACd,IAAI,CAAC,iBAAiB,CAAC,aAAa,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;4BAChD,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,GAAG,EAAE;gCACnB,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC;4BAC9B,CAAC,CAAC,CAAC;yBACJ;qBACF;oBAAC,MAAM;wBACN,OAAO,CAAC,GAAG,CAAC,6BAA6B,CAAC,CAAC;qBAC5C;gBACH,CAAC,CAAC,CAAC;gBAEH,gBAAgB;gBAChB,yDAAyD;gBACzD,iGAAiG;gBACjG,IAAI,CAAC,YAAY,GAAG,WAAW,CAC7B,IAAI,CAAC,kBAAkB,CAAC,IAAI,CAAC,IAAI,CAAC,EAClC,EAAE,CACH,CAAC;gBAEF;;mBAEG;gBACH,IAAI,CAAC,GAAG,CAAC,EAAE,CACT,sBAAsB,EACtB,UAAU,EAAE,EAAE,IAAI;oBAChB,8DAA8D;oBAC9D,IAAI,CAAC,gBAAgB,GAAG,IAAI,CAAC;gBAC/B,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CACb,CAAC;gBAEF,IAAI,CAAC,GAAG,CAAC,EAAE,CACT,oBAAoB,EACpB,UAAU,OAAO,EAAE,OAAO;oBACxB,mEAAmE;oBACnE,IAAI,OAAO,KAAK,oBAAoB,EAAE;wBACpC,IAAI,CAAC,gBAAgB,GAAG,KAAK,CAAC;qBAC/B;gBACH,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CACb,CAAC;gBAEF,IAAI,CAAC,GAAG,CAAC,EAAE,CACT,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC,KAAK,CAAC,KAAK,EAC1B,UAAU,QAAQ,EAAE,QAAQ;oBAC1B,4CAA4C;oBAC5C,6CAA6C;oBAC7C,IAAI,CAAC,iBAAiB,GAAG,KAAK,CAAC;oBAC/B,IAAI,CAAC,oBAAoB,EAAE,CAAC;gBAC9B,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CACb,CAAC;gBACF,IAAI,CAAC,GAAG,CAAC,EAAE,CACT,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC,KAAK,CAAC,IAAI,EACzB,UAAU,SAAS,EAAE,OAAO;oBAC1B,+CAA+C;oBAC/C,sDAAsD;oBACtD,IAAI,CAAC,iBAAiB,GAAG,IAAI,CAAC;gBAChC,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CACb,CAAC;gBAEF,uCAAuC;gBACvC,MAAM,CAAC,WAAW,CAAC,GAAG,MAAM,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC,aAAa,CAAC,CAAC,CAAC,CAAC;gBAC5D,MAAM,IAAI,GAAG,WAAW,CAAC,OAAO,EAAE,CAAC;gBACnC,IAAI,CAAC,KAAK,EAAE,CAAC;gBACb,IAAI,CAAC,YAAY,GAAG,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC;gBAEtC,kEAAkE;gBAClE,6CAA6C;gBAC7C,kDAAkD;gBAClD,uCAAuC;gBACvC,sCAAsC;gBACtC,sDAAsD;gBACtD,MAAM;gBAGN,8DAA8D;gBAC9D,8CAA8C;gBAC9C,gCAAgC;gBAChC,wDAAwD;gBACxD,EAAE;gBACF,6BAA6B;gBAC7B,oDAAoD;gBACpD,EAAE;gBACF,yBAAyB;gBACzB,YAAY;gBACZ,qCAAqC;gBACrC,4EAA4E;gBAC5E,yCAAyC;gBACzC,+BAA+B;gBAC/B,wBAAwB;gBACxB,4CAA4C;gBAC5C,eAAe;gBACf,aAAa;gBACb,YAAY;gBACZ,oBAAoB;gBACpB,sDAAsD;gBACtD,0BAA0B;gBAC1B,QAAQ;gBACR,qBAAqB;gBACrB,gBAAgB;gBAChB,QAAQ;gBACR,EAAE;gBACF,yBAAyB;gBACzB,YAAY;gBACZ,wFAAwF;gBACxF,oBAAoB;gBACpB,sDAAsD;gBACtD,0BAA0B;gBAC1B,QAAQ;gBACR,qBAAqB;gBACrB,gBAAgB;gBAChB,QAAQ;gBACR,EAAE;gBACF,mCAAmC;gBACnC,kCAAkC;gBAClC,qCAAqC;gBACrC,oCAAoC;gBACpC,wCAAwC;gBACxC,iCAAiC;gBACjC,6BAA6B;gBAC7B,gDAAgD;gBAChD,uCAAuC;gBACvC,+DAA+D;gBAC/D,+BAA+B;gBAC/B,mBAAmB;gBACnB,+BAA+B;gBAC/B,8DAA8D;gBAC9D,YAAY;gBACZ,EAAE;gBACF,wDAAwD;gBACxD,aAAa;gBACb,oCAAoC;gBACpC,oDAAoD;gBACpD,mCAAmC;gBACnC,sBAAsB;gBACtB,EAAE;gBACF,yEAAyE;gBACzE,gDAAgD;gBAChD,4DAA4D;gBAC5D,iDAAiD;gBACjD,+BAA+B;gBAC/B,sBAAsB;gBACtB,aAAa;gBACb,UAAU;gBACV,UAAU;gBACV,gCAAgC;gBAChC,kBAAkB;gBAClB,iDAAiD;gBACjD,6HAA6H;gBAC7H,MAAM;gBACN,MAAM;gBAEN,OAAO,CAAC,IAAI,CAAC,CAAC;YAChB,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CACb,CAAC;QACJ,CAAC,CAAC,CAAC;IACL,CAAC;IAGD,cAAc;QACZ,IAAI,CAAC,iBAAiB,GAAG,IAAI,CAAC;IAChC,CAAC;IAED,aAAa,CAAC,KAA0C;QACtD,IAAI,CAAC,GAAG,KAAK,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC;QAC3B,IAAI,CAAC,GAAG,KAAK,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC;QAC3B,IAAI,CAAC,GAAG,KAAK,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC;QAE3B,OAAO,QAAQ,CAAC,QAAQ,CAAC,QAAQ,CAAC,IAAI,CAAC;IACzC,CAAC;IAED,EAAE;IACF,6CAA6C;IAC7C,EAAE;IACF;;OAEG;IACH,6BAA6B;QAC3B,IAAI,IAAI,CAAC,WAAW,CAAC,MAAM,GAAG,CAAC,EAAE;YAC/B,MAAM,YAAY,GAAG,IAAI,CAAC,WAAW,CAAC,MAAM,CAAC;YAC7C,IAAI,CAAC,oBAAoB,GAAG,EAAE,CAAC;YAC/B,KAAK,IAAI,KAAK,GAAG,CAAC,EAAE,KAAK,GAAG,YAAY,EAAE,KAAK,IAAI,CAAC,EAAE;gBACpD,MAAM,QAAQ,GAAG,2BAA2B,CAC1C,IAAI,CAAC,WAAW,CAAC,KAAK,GAAG,CAAC,CAAC,EAC3B,IAAI,CAAC,WAAW,CAAC,KAAK,CAAC,CACxB,CAAC;gBACF,IAAI,CAAC,oBAAoB,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;aAC1C;YACD,IAAI,CAAC,cAAc,EAAE,CAAC,IAAI,CAAC,CAAC,GAAG,EAAE,EAAE;gBACjC,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,CAAC,SAAS,IAAI,CAAC,cAAc,kBAAkB,CAAC,CAAC,CAAC;YACzE,CAAC,CAAC,CAAC;SACJ;IACH,CAAC;IAED,gBAAgB;QACd,OAAO,IAAI,CAAC,oBAAoB,CAAC;IACnC,CAAC;IAED;;;OAGG;IACH,cAAc;QACZ,OAAO,IAAI,CAAC,GAAG,CAAC,QAAQ,CAAC,cAAc,CACrC,IAAI,CAAC,UAAU,EACf,IAAI,CAAC,UAAU,CAChB,CAAC,IAAI,CACJ,UAAU,aAAkB;YAC1B,gBAAgB;YAChB,IAAI,CAAC,iBAAiB,GAAG,aAAa,CAAC;YACvC,OAAO,OAAO,CAAC,OAAO,EAAE,CAAC;QAC3B,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CACb,CAAC;IACJ,CAAC;IAED,gBAAgB;QACd,OAAO,IAAI,CAAC,iBAAiB,CAAC;IAChC,CAAC;IAED,kBAAkB;QAChB,MAAM,IAAI,GAAG;YACX,OAAO,EAAE,IAAI,CAAC,WAAW;YACzB,KAAK,EAAE,IAAI,CAAC,UAAU,CAAC,KAAK;SAC7B,CAAC;QACF,OAAO,IAAI,CAAC;IACd,CAAC;IAED,EAAE;IACF,8BAA8B;IAC9B,EAAE;IACF;;OAEG;IACH,kBAAkB;QAChB,IACE,IAAI,CAAC,eAAe,KAAK,kBAAkB,CAAC,OAAO;YACnD,IAAI,CAAC,iBAAiB;YACtB,IAAI,CAAC,cAAc;YACnB,IAAI,CAAC,eAAe,CAClB,IAAI,CAAC,cAAc,CAAC,QAAQ,EAC5B,IAAI,CAAC,yBAAyB,CAC/B,GAAG,EAAE,EACN;YACA,IAAI,CAAC,aAAa,CAAC,KAAK,CAAC,OAAO,GAAG,MAAM,CAAC;YAE1C,MAAM,IAAI,GAAG;gBACX,CAAC,EAAE,IAAI,CAAC,SAAS,CAAC,WAAW;gBAC7B,CAAC,EAAE,IAAI,CAAC,SAAS,CAAC,YAAY;aAC/B,CAAC;YACF,MAAM,MAAM,GAAG,IAAI,CAAC,GAAG,CAAC,UAAU,CAAC,aAAa,CAC9C,IAAI,CAAC,cAAc,CAAC,QAAQ,EAC5B,IAAI,CAAC,UAAU,EACf,IAAI,CACL,CAAC;YACF,IAAI,CAAC,aAAa,CAAC,KAAK,CAAC,IAAI,GAAG,GAC9B,MAAM,CAAC,CAAC,GAAG,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,GAAG,EACnC,IAAI,CAAC;YACL,IAAI,CAAC,aAAa,CAAC,KAAK,CAAC,GAAG,GAAG,GAAG,MAAM,CAAC,CAAC,GAAG,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,GAAG,EAAE,IAAI,CAAC;YAC1E,IAAI,CAAC,yBAAyB,GAAG;gBAC/B,GAAG,IAAI,CAAC,cAAc,CAAC,QAAQ;aAChC,CAAC;SACH;aAAM;YACL,IAAI,CAAC,aAAa,CAAC,KAAK,CAAC,OAAO,GAAG,OAAO,CAAC;SAC5C;IACH,CAAC;IAED;;;OAGG;IACH,oBAAoB,CAAC,SAAiB;QACpC,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC,YAAY,CAAC,SAAS,EAAE;YACnC,cAAc,EAAE;gBACd,CAAC,EAAE,IAAI,CAAC,cAAc,CAAC,QAAQ,CAAC,CAAC,GAAG,CAAC;gBACrC,CAAC,EAAE,IAAI,CAAC,cAAc,CAAC,QAAQ,CAAC,CAAC,GAAG,CAAC;gBACrC,CAAC,EAAE,IAAI,CAAC,cAAc,CAAC,QAAQ,CAAC,CAAC,GAAG,CAAC;aACtC;YACD,UAAU,EAAE;gBACV,CAAC,EAAE,IAAI,CAAC,cAAc,CAAC,MAAM,CAAC,CAAC,GAAG,GAAG;gBACrC,CAAC,EAAE,IAAI,CAAC,cAAc,CAAC,MAAM,CAAC,CAAC,GAAG,GAAG;gBACrC,CAAC,EAAE,IAAI,CAAC,cAAc,CAAC,MAAM,CAAC,CAAC,GAAG,GAAG;aACtC;SACF,CAAC,CAAC;IACL,CAAC;IAED;;;;;OAKG;IACH,eAAe,CAAC,IAAS,EAAE,IAAS;QAClC,MAAM,IAAI,GAAG;YACX,CAAC,EAAE,IAAI,CAAC,SAAS,CAAC,WAAW;YAC7B,CAAC,EAAE,IAAI,CAAC,SAAS,CAAC,YAAY;SAC/B,CAAC;QACF,MAAM,OAAO,GAAG,IAAI,CAAC,GAAG,CAAC,UAAU,CAAC,aAAa,CAC/C,IAAI,EACJ,IAAI,CAAC,UAAU,EACf,IAAI,CACL,CAAC;QACF,MAAM,OAAO,GAAG,IAAI,CAAC,GAAG,CAAC,UAAU,CAAC,aAAa,CAC/C,IAAI,EACJ,IAAI,CAAC,UAAU,EACf,IAAI,CACL,CAAC;QACF,OAAO,IAAI,CAAC,IAAI,CACd,CAAC,OAAO,CAAC,CAAC,GAAG,OAAO,CAAC,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC,GAAG,OAAO,CAAC,CAAC,CAAC,IAAI,CAAC,CAC5D,CAAC;IACJ,CAAC;IAED,EAAE;IACF,0CAA0C;IAC1C,EAAE;IACF;;;OAGG;IACH,KAAK,CAAC,kBAAkB,CAAC,aAA4B;QACnD,IAAI,CAAC,IAAI,CAAC,cAAc;YAAE,OAAO;QACjC,IAAI,CAAC,iBAAiB,GAAG,MAAM,IAAI,CAAC,oBAAoB,CAAC,aAAa,CAAC,CAAC;QACxE,OAAO,CAAC,GAAG,CAAC,mBAAmB,EAAE,IAAI,CAAC,iBAAiB,CAAC,CAAC;QACzD,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC,QAAQ,CAAC,IAAI,CAAC,iBAAiB,EAAE,eAAe,CAAC,CAAC;QAC/D,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC,WAAW,CAAC,IAAI,CAAC,iBAAiB,EAAE,CAAC,CAAC,CAAC;IACtD,CAAC;IAED;;;;;OAKG;IACH,KAAK,CAAC,oBAAoB,CACxB,aAA4B;QAE5B,IAAI,OAAO,CAAC;QACZ,IAAI,CAAC,IAAI,CAAC,GAAG,EAAE;YACb,OAAO,IAAI,CAAC;SACb;QACD,IAAI;YACF,OAAO,GAAG,MAAM,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC,GAAG,CAAC,aAAa,CAAC,OAAO,EAAE,CAAC,CAAC;SAC3D;QAAC,OAAO,KAAK,EAAE;YACd,OAAO,CAAC,GAAG,CACT,kCAAkC,EAClC,KAAK,EACL,aAAa,CAAC,OAAO,EAAE,CACxB,CAAC;SACH;QACD,IAAI,OAAO,EAAE;YACX,MAAM,WAAW,GAAG,OAAO,CAAC,CAAC,CAAC,CAAC;YAC/B,kEAAkE;YAClE,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;YACpC,IAAI,CAAC,eAAe,CAAC,GAAG,CAAC,WAAW,EAAE,aAAa,CAAC,CAAC;YACrD,OAAO,WAAW,CAAC;SACpB;QACD,OAAO,IAAI,CAAC;IACd,CAAC;IAED;;OAEG;IACH,oBAAoB;QAClB,IAAI,CAAC,kBAAkB,CAAC,kBAAkB,CAAC,OAAO,CAAC,CAAC;QACpD,MAAM,OAAO,GAAG,IAAI,CAAC,UAAU,EAAE,CAAC;QAClC,IAAI,OAAO,EAAE;YACX,MAAM,YAAY,GAAG,IAAI,CAAC,eAAe;iBACtC,GAAG,CAAC,OAAO,CAAC;iBACZ,qBAAqB,EAAE,CAAC;YAC3B,IAAI,CAAC,gCAAgC,CAAC,YAAY,CAAC,CAAC;SACrD;IACH,CAAC;IAED;;;;OAIG;IACH,KAAK,CAAC,sBAAsB,CAAC,WAAmB,EAAE,QAAgB;QAChE,MAAM,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC,eAAe,CAAC,QAAQ,WAAW,EAAE,EAAE,QAAQ,CAAC,CAAC;QACtE,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC,QAAQ,CAAC,WAAW,EAAE,QAAQ,WAAW,EAAE,CAAC,CAAC;IAC5D,CAAC;IAED;;;;OAIG;IACH,KAAK,CAAC,sBAAsB,CAAC,WAAmB,EAAE,QAAgB;QAChE,IAAI;YACF,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC,QAAQ,CAAC,WAAW,EAAE,QAAQ,CAAC,CAAC;SAC9C;QAAC,OAAO,CAAC,EAAE;YACV,OAAO,CAAC,GAAG,CACT,gCAAgC,EAChC,QAAQ,EACR,qBAAqB,CACtB,CAAC;SACH;IACH,CAAC;IAED;;;;;OAKG;IACH,KAAK,CAAC,oBAAoB,CACxB,WAAmB,EACnB,aAA4B;QAE5B,IAAI,IAAI,CAAC,QAAQ,KAAK,QAAQ,CAAC,KAAK,EAAE;YACpC,MAAM,IAAI,GAAG,aAAa,CAAC,SAAS,EAAE,CAAC;YACvC,IAAI,QAAQ,GAAG,WAAW,CAAC;YAC3B,IAAI,IAAI,CAAC,SAAS,EAAE;gBAClB,QAAQ,GAAG,GAAG,QAAQ,YAAY,CAAC;aACpC;iBAAM;gBACL,QAAQ,GAAG,GAAG,QAAQ,cAAc,CAAC;aACtC;YACD,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC,QAAQ,CAAC,WAAW,EAAE,QAAQ,CAAC,CAAC;YAC7C,OAAO;SACR;QACD,MAAM,aAAa,GAAG,eAAe,CAAC,aAAa,CAAC,OAAO,EAAE,CAAC,CAAC;QAC/D,IAAI,OAAO,GAAG,IAAI,CAAC,MAAM,CAAC,SAAS,CAAC,mBAAmB,CAAC;QACxD,IAAI,QAAQ,GAAG,QAAQ,aAAa,EAAE,CAAC;QACvC,MAAM,GAAG,GAAG,aAAa,CAAC,MAAM,EAAE,CAAC;QACnC,IAAI,GAAG,IAAI,GAAG,CAAC,OAAO,EAAE;YACtB,MAAM,OAAO,GAAG,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC;YACxC,IAAI,aAAa,CAAC,OAAO,EAAE,KAAK,OAAO,CAAC,IAAI,EAAE;gBAC5C,MAAM,OAAO,GAAG,aAAa,CAAC,SAAS,EAAE,CAAC;gBAC1C,IAAI,OAAO,CAAC,IAAI,KAAK,WAAW,CAAC,cAAc,EAAE;oBAC/C,OAAO,CAAC,GAAG,GAAG,IAAI,CAAC,UAAU,CAAC,yBAAyB,CAAC,OAAO,EAAE,GAAG,CAAC,CAAC;iBACvE;aACF;YACD,IAAI,OAAO,CAAC,GAAG,EAAE;gBACf,MAAM,UAAU,GAAG,MAAM,IAAI,CAAC,UAAU,CAAC,sBAAsB,CAC7D,OAAO,CAAC,GAAG,CACZ,CAAC;gBACF,IAAI,UAAU,EAAE;oBACd,IAAI;wBACF,QAAQ,GAAG,QAAQ,aAAa,IAAI,WAAW,IAAI,aAAa,CAAC,eAAe,EAAE,CAAC;wBACnF,MAAM,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC,eAAe,CAAC,QAAQ,EAAE,UAAU,CAAC,CAAC;wBAC3D,aAAa,CAAC,eAAe,IAAI,CAAC,CAAC;qBACpC;oBAAC,MAAM;wBACN,OAAO,CAAC,GAAG,CAAC,oBAAoB,CAAC,CAAC;qBACnC;iBACF;aACF;YACD,IAAI,OAAO,CAAC,OAAO,EAAE;gBACnB,OAAO,GAAG,OAAO,CAAC,OAAO,CAAC;aAC3B;SACF;QACD,IAAI;YACF,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC,QAAQ,CAAC,WAAW,EAAE,QAAQ,CAAC,CAAC;YAC7C,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC,WAAW,CAAC,WAAW,EAAE,OAAO,CAAC,CAAC;SAChD;QAAC,OAAO,KAAK,EAAE;YACd,OAAO,CAAC,GAAG,CAAC,2BAA2B,EAAE,KAAK,CAAC,CAAC;SACjD;IACH,CAAC;IAED;;OAEG;IACH,WAAW;QACT,UAAU,CAAC,GAAG,EAAE;YACd,MAAM,OAAO,GAAG,IAAI,CAAC,UAAU,EAAE,CAAC;YAClC,IAAI,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC;QACxB,CAAC,EAAE,IAAI,CAAC,CAAC;IACX,CAAC;IAED;;;;OAIG;IACH,KAAK,CAAC,OAAO,CAAC,WAAmB;QAC/B,IAAI,WAAW,KAAK,EAAE;YAAE,OAAO;QAC/B,IAAI;YACF,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;YACjC,MAAM,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC,OAAO,CAAC,SAAS,CACpC,CAAC,YAAY,EAAE,EAAE,CAAC,YAAY,KAAK,EAAE,CACtC,CAAC;YACF,MAAM,IAAI,CAAC,GAAG,CAAC,SAAS,CAAC,aAAa,CACpC,WAAW,EACX,IAAI,CAAC,GAAG,CAAC,SAAS,CAAC,UAAU,CAAC,GAAG,CAClC,CAAC;SACH;QAAC,OAAO,KAAK,EAAE;YACd,OAAO,CAAC,GAAG,CAAC,wBAAwB,EAAE,KAAK,CAAC,CAAC;SAC9C;IACH,CAAC;IAED;;;;;OAKG;IACH,KAAK,CAAC,8BAA8B,CAClC,WAAmB,EACnB,SAAuB,IAAI,EAC3B,UAAmB,IAAI;QAEvB,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC,aAAa,CACxB,WAAW,EACX,IAAI,CAAC,eAAe,CAAC,GAAG,CAAC,WAAW,CAAC,CAAC,OAAO,EAAE,CAChD,CAAC;QACF,IAAI,MAAM,IAAI,OAAO,EAAE;YACrB,MAAM,IAAI,CAAC,eAAe,CAAC,OAAO,EAAE,MAAM,EAAE,WAAW,CAAC,CAAC;SAC1D;QACD,MAAM,IAAI,CAAC,oBAAoB,CAC7B,WAAW,EACX,IAAI,CAAC,eAAe,CAAC,GAAG,CAAC,WAAW,CAAC,CACtC,CAAC;IACJ,CAAC;IAED;;;;;OAKG;IACH,KAAK,CAAC,0BAA0B,CAC9B,WAAmB,EACnB,MAAW,EACX,OAAgB;QAEhB,MAAM,IAAI,CAAC,eAAe,CAAC,OAAO,EAAE,MAAM,EAAE,WAAW,CAAC,CAAC;IAC3D,CAAC;IAED;;;OAGG;IACH,qBAAqB,CAAC,WAAmB;QACvC,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC,MAAM,CAAC,WAAW,CAAC,CAAC;QACjC,IAAI,CAAC,eAAe,CAAC,MAAM,CAAC,WAAW,CAAC,CAAC;IAC3C,CAAC;IAED;;OAEG;IACH,mBAAmB;QACjB,MAAM,WAAW,GAAG,IAAI,CAAC,YAAY,CAAC,GAAG,EAAE,CAAC;QAC5C,IAAI,WAAW,EAAE;YACf,IAAI,CAAC,qBAAqB,CAAC,WAAW,CAAC,CAAC;SACzC;IACH,CAAC;IAED;;;;;;;OAOG;IACH,KAAK,CAAC,sBAAsB,CAC1B,OAAgB,EAChB,MAAoB,EACpB,GAAQ;QAER,8BAA8B;QAC9B,MAAM,EAAC,GAAG,EAAE,KAAK,EAAC,GAAG,IAAI,CAAC,mBAAmB,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC;QACzD,IAAI,GAAG,EAAE;YACP,oCAAoC;YACpC,OAAO;SACR;QACD,MAAM,aAAa,GAAkB,IAAI,aAAa,CAAC,OAAO,CAAC,CAAC;QAChE,aAAa,CAAC,SAAS,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC;QACzC,IAAI,GAAG,CAAC,UAAU,EAAE;YAClB,aAAa,CAAC,WAAW,CAAC,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,UAAU,CAAC,CAAC,CAAC;SACvD;QACD,IAAI,GAAG,CAAC,QAAQ,EAAE;YAChB,MAAM,WAAW,GAAG,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,QAAQ,CAAC,CAAC;YAC7C,IAAI,WAAW,CAAC,MAAM,EAAE;gBACtB,aAAa,CAAC,SAAS,CAAC,WAAW,CAAC,MAAM,CAAC,CAAC;aAC7C;iBAAM;gBACL,aAAa,CAAC,SAAS,CAAC,EAAC,CAAC,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,IAAI,EAAE,CAAC,EAAE,CAAC,EAAC,CAAC,CAAC;aACjD;SACF;QACD,IAAI,GAAG,CAAC,iBAAiB,EAAE;YACzB,aAAa,CAAC,UAAU,CAAC,GAAG,CAAC,iBAAiB,CAAC,CAAC;SACjD;QACD,aAAa,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC;QAE1B,MAAM,YAAY,GAAG,MAAM,IAAI,CAAC,oBAAoB,CAAC,aAAa,CAAC,CAAC;QACpE,IAAI,YAAY,IAAI,IAAI,CAAC,GAAG,EAAE;YAC5B,MAAM,IAAI,CAAC,oBAAoB,CAAC,YAAY,EAAE,aAAa,CAAC,CAAC;YAC7D,MAAM,IAAI,CAAC,eAAe,CAAC,OAAO,EAAE,MAAM,EAAE,YAAY,CAAC,CAAC;SAC3D;IACH,CAAC;IAED;;;;;OAKG;IACH,KAAK,CAAC,eAAe,CAAC,OAAgB,EAAE,MAAoB,EAAE,KAAa;QACzE,IAAI,IAAI,GAAG,MAAM,IAAI,CAAC,UAAU,CAAC,eAAe,CAAC,OAAO,EAAE,MAAM,CAAC,CAAC;QAClE,IAAI,IAAI,KAAK,EAAE,IAAI,IAAI,CAAC,GAAG,EAAE;YAC3B,MAAM,SAAS,GAAG,IAAI,CAAC,UAAU,CAAC,eAAe,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC;YACnE,IAAI,IAAI,GAAG,SAAS,EAAE,CAAC;YACvB,kCAAkC;YAClC,MAAM,CAAC,SAAS,EAAE,SAAS,CAAC,GAAG,MAAM,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC,eAAe,CAAC,IAAI,CAAC,CAAC;YAExE,qCAAqC;YACrC,IAAI,YAAY,GAAG,IAAI,CAAC,eAAe,CAAC,KAAK,CAAC,CAAC;YAC/C,IAAI,YAAY,EAAE;gBAChB,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC,MAAM,CAAC,KAAK,EAAE,YAAY,CAAC,CAAC;aAC1C;YACD,IAAI,CAAC,eAAe,CAAC,KAAK,CAAC,GAAG,SAAS,CAAC;YAExC,8BAA8B;YAC9B,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC,MAAM,CAAC,KAAK,EAAE,SAAS,CAAC,CAAC;YACtC,gCAAgC;YAChC,6CAA6C;YAC7C,IAAI,CAAC,IAAI,CAAC,cAAc,EAAE;gBACxB,IAAI,CAAC,cAAc,GAAG,IAAI,CAAC;gBAC3B,MAAM,UAAU,GAAG,CAAC,SAAiB,EAAE,EAAE;oBACvC,iDAAiD;oBACjD,IAAI,CAAC,UAAU,CAAC,kBAAkB,CAAC,SAAS,CAAC,CAAC;gBAChD,CAAC,CAAC;gBAEF,MAAM,UAAU,GAAG,CAAC,cAAsB,EAAE,EAAE;oBAC5C,sDAAsD;oBACtD,IAAI,CAAC,UAAU,CAAC,kBAAkB,CAAC,cAAc,CAAC,CAAC;gBACrD,CAAC,CAAC;gBACF,MAAM,UAAU,GAAG,CAAC,GAAW,EAAE,EAAE;oBACjC,IAAI,CAAC,UAAU,CAAC,kBAAkB,CAAC,GAAG,CAAC,CAAC;gBAC1C,CAAC,CAAC;gBACF,SAAS,CAAC,EAAE,CACV,SAAS,CAAC,YAAY,EACtB,IAAI,CAAC,UAAU,CAAC,mBAAmB,CAAC,IAAI,CAAC,IAAI,CAAC,UAAU,CAAC,CAC1D,CAAC;gBACF,SAAS,CAAC,EAAE,CACV,SAAS,CAAC,YAAY,EACtB,IAAI,CAAC,UAAU,CAAC,mBAAmB,CAAC,IAAI,CAAC,IAAI,CAAC,UAAU,CAAC,CAC1D,CAAC;gBACF,SAAS,CAAC,EAAE,CAAC,SAAS,CAAC,WAAW,EAAE,UAAU,CAAC,CAAC;gBAChD,SAAS,CAAC,EAAE,CAAC,SAAS,CAAC,WAAW,EAAE,UAAU,CAAC,CAAC;gBAChD,SAAS,CAAC,EAAE,CAAC,SAAS,CAAC,WAAW,EAAE,UAAU,CAAC,CAAC;gBAChD,SAAS,CAAC,EAAE,CACV,SAAS,CAAC,SAAS,EACnB,IAAI,CAAC,UAAU,CAAC,gBAAgB,CAAC,IAAI,CAAC,IAAI,CAAC,UAAU,CAAC,CACvD,CAAC;gBACF,SAAS,CAAC,EAAE,CACV,SAAS,CAAC,aAAa,EACvB,IAAI,CAAC,UAAU,CAAC,oBAAoB,CAAC,IAAI,CAAC,IAAI,CAAC,UAAU,CAAC,CAC3D,CAAC;aACH;SACF;aAAM;YACL,iGAAiG;YACjG,MAAM,EAAC,OAAO,EAAE,cAAc,EAAC,GAC7B,IAAI,CAAC,UAAU,CAAC,wBAAwB,CAAC,MAAM,CAAC,CAAC;YACnD,IAAI,OAAO,EAAE;gBACX,qBAAqB;gBACrB,MAAM,CAAC,YAAY,CAAC,GAAG,MAAM,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC,kBAAkB,CAC1D,OAAO,CAAC,YAAY,CACrB,CAAC;gBACF,SAAS;gBACT,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC,MAAM,CAAC,KAAK,EAAE,YAAY,CAAC,CAAC;gBACzC,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC,aAAa,CAAC,KAAK,EAAE;oBAChC,KAAK,EAAE,MAAM,CAAC,KAAK;oBACnB,WAAW,EAAE,cAAc;iBAC5B,CAAC,CAAC;aACJ;SACF;IACH,CAAC;IAED,KAAK,CAAC,4BAA4B;QAChC,MAAM,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC,MAAM,CAAC,GAAG,IAAI,CAAC,YAAY,CAAC,CAAC;QAChD,IAAI,CAAC,YAAY,GAAG,EAAE,CAAC;QACvB,IAAI,CAAC,eAAe,CAAC,KAAK,EAAE,CAAC;IAC/B,CAAC;IAED;;;OAGG;IACH,KAAK,CAAC,wBAAwB,CAAC,SAAiB;QAC9C,MAAM,WAAW,GAAG,IAAI,CAAC,mBAAmB,CAAC,SAAS,CAAC,CAAC,GAAG,CAAC;QAC5D,IAAI,WAAW,EAAE;YACf,IAAI;gBACF,MAAM,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC,MAAM,CAAC,WAAW,CAAC,CAAC;gBACvC,IAAI,CAAC,eAAe,CAAC,MAAM,CAAC,WAAW,CAAC,CAAC;gBACzC,MAAM,KAAK,GAAG,IAAI,CAAC,YAAY,CAAC,OAAO,CAAC,WAAW,CAAC,CAAC;gBACrD,IAAI,CAAC,YAAY,CAAC,MAAM,CAAC,KAAK,EAAE,CAAC,CAAC,CAAC;aACpC;YAAC,OAAO,KAAK,EAAE;gBACd,OAAO,CAAC,GAAG,CAAC,mBAAmB,EAAE,WAAW,EAAE,KAAK,CAAC,CAAC;aACtD;SACF;IACH,CAAC;IAED;;;OAGG;IACH,KAAK,CAAC,cAAc,CAAC,IAAY;QAC/B,IAAI,CAAC,oBAAoB,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC,KAAK,CAAC,OAAO,GAAG,KAAK,CAAC;IAC5D,CAAC;IAED,mCAAmC,CAAC,IAAY;QAC5C,IAAI,GAAG,GAAG,IAAI,CAAC,oBAAoB,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;QAC9C,IAAG,CAAC,GAAG,EAAC;YACN,OAAO,CAAC,GAAG,CAAC,mBAAmB,CAAC,CAAC;YACjC,OAAO,IAAI,CAAC;SACb;QACD,0CAA0C;QAC1C,IAAG,GAAG,CAAC,KAAK,CAAC,IAAI,IAAI,IAAI,EAAC;YACxB,OAAO,CAAC,GAAG,CAAC,qBAAqB,CAAC,CAAC;YACnC,GAAG,CAAC,KAAK,CAAC,IAAI,GAAG,IAAI,CAAC,CAAC,yBAAyB;SACjD;QAED,OAAO,GAAG,CAAC;IACf,CAAC;IAED;;;;OAIG;IACH,KAAK,CAAC,6BAA6B,CAAC,OAAgB,EAAE,OAAqB;QACzE,MAAM,aAAa,GAAG,IAAI,aAAa,CAAC,OAAO,CAAC,CAAC;QACjD,0CAA0C;QAC1C,MAAM,CAAC,GAAG,CAAC,GAAG,OAAO,CAAC,IAAI,CAAC,KAAK,CAAC;QACjC,IAAI,GAAG,IAAI,GAAG,CAAC,UAAU,EAAE;YACzB,aAAa,CAAC,WAAW,CAAC,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,UAAU,CAAC,CAAC,CAAC;YACtD,aAAa,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC,qCAAqC;SACjE;QACD,aAAa,CAAC,UAAU,CAAC,IAAI,CAAC,UAAU,CAAC,KAAK,CAAC,CAAC;QAChD,aAAa,CAAC,WAAW,CAAC,IAAI,CAAC,UAAU,CAAC,QAAQ,CAAC,CAAC;QACpD,aAAa,CAAC,SAAS,CAAC,OAAO,EAAE,OAAO,CAAC,CAAC;QAC1C,IAAI,CAAC,iBAAiB,GAAG,MAAM,IAAI,CAAC,oBAAoB,CAAC,aAAa,CAAC,CAAC;QACxE,IAAI,CAAC,kBAAkB,CAAC,kBAAkB,CAAC,WAAW,CAAC,CAAC;QACxD,MAAM,IAAI,CAAC,kBAAkB,CAAC,aAAa,CAAC,CAAC;IAC/C,CAAC;IAED;;;OAGG;IACH,KAAK,CAAC,sBAAsB,CAAC,OAAgB;QAC3C,MAAM,aAAa,GAAG,IAAI,aAAa,CAAC,OAAO,CAAC,CAAC;QACjD,aAAa,CAAC,UAAU,CAAC,IAAI,CAAC,UAAU,CAAC,KAAK,CAAC,CAAC;QAChD,aAAa,CAAC,WAAW,CAAC,IAAI,CAAC,UAAU,CAAC,QAAQ,CAAC,CAAC;QACpD,IAAI,CAAC,kBAAkB,CAAC,kBAAkB,CAAC,MAAM,CAAC,CAAC;QACnD,MAAM,IAAI,CAAC,kBAAkB,CAAC,aAAa,CAAC,CAAC;IAC/C,CAAC;IAED;;OAEG;IACH,qBAAqB;QACnB,IAAI,CAAC,mBAAmB,EAAE,CAAC;QAC3B,IAAI,CAAC,kBAAkB,CAAC,kBAAkB,CAAC,OAAO,CAAC,CAAC;IACtD,CAAC;IAED,eAAe,CAAC,YAA6B;QAC3C,IAAI,CAAC,YAAY,GAAG,YAAY,CAAC;IACnC,CAAC;IAED,eAAe;QACb,OAAO,IAAI,CAAC,YAAY,CAAC;IAC3B,CAAC;IAED;;;OAGG;IACH,gCAAgC,CAAC,IAAyB;QACxD,IAAI,CAAC,iBAAiB,CAAC,aAAa,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QAChD,MAAM,OAAO,GAAG,IAAI,CAAC,UAAU,EAAE,CAAC;QAClC,QAAQ,IAAI,EAAE;YACZ,KAAK,mBAAmB,CAAC,aAAa;gBACpC,IAAI,CAAC,WAAW,EAAE,CAAC;gBACnB,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,CAAC,SAAS,IAAI,CAAC,cAAc,YAAY,CAAC,CAAC,CAAC;gBACjE,MAAM;YACR,KAAK,mBAAmB,CAAC,UAAU;gBACjC,IAAI,CAAC,WAAW,EAAE,CAAC;gBACnB,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,CAAC,SAAS,IAAI,CAAC,cAAc,aAAa,CAAC,CAAC,CAAC;gBAClE,MAAM;YACR,KAAK,mBAAmB,CAAC,YAAY;gBACnC,IAAI,CAAC,WAAW,EAAE,CAAC;gBACnB,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,CAAC,SAAS,IAAI,CAAC,cAAc,eAAe,CAAC,CAAC,CAAC;gBACpE,MAAM;YACR,KAAK,mBAAmB,CAAC,QAAQ;gBAC/B,IAAI,CAAC,WAAW,EAAE,CAAC;gBACnB,IAAI,GAAW,CAAC;gBAChB,IAAI,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,QAAQ,CAAC,GAAG,CAAC,EAAE;oBACjC,GAAG,GAAG,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC,EAAE,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC,CAAC;iBAClE;qBAAM;oBACL,GAAG,GAAG,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC;iBACvB;gBACD,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,CAAC,GAAG,GAAG,cAAc,CAAC,CAAC,CAAC;gBAC7C,MAAM;YACR,KAAK,mBAAmB,CAAC,QAAQ;gBAC/B,IAAI,CAAC,WAAW,EAAE,CAAC;gBACnB,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,CAAC,SAAS,IAAI,CAAC,cAAc,cAAc,CAAC,EAAE;oBACjE,WAAW,EAAE,EAAC,MAAM,EAAE,IAAI,EAAC;iBAC5B,CAAC,CAAC;gBACH,MAAM;YACR,KAAK,mBAAmB,CAAC,QAAQ;gBAC/B,IAAI,CAAC,WAAW,EAAE,CAAC;gBACnB,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,CAAC,SAAS,IAAI,CAAC,cAAc,WAAW,CAAC,CAAC,CAAC;gBAChE,MAAM;YACR,KAAK,mBAAmB,CAAC,iBAAiB;gBACxC,IAAI,OAAO,EAAE;oBACX,6BAA6B;oBAC7B,IAAI,CAAC,MAAM,CAAC,QAAQ,CAClB,CAAC,SAAS,IAAI,CAAC,cAAc,aAAa,IAAI,CAAC,YAAY,CAAC,EAAE,EAAE,CAAC,EACjE,EAAC,WAAW,EAAE,EAAC,WAAW,EAAE,IAAI,EAAC,EAAC,CACnC,CAAC;iBACH;gBACD,MAAM;YACR,KAAK,mBAAmB,CAAC,eAAe;gBACtC,IAAI,OAAO,EAAE;oBACX,IAAI,CAAC,MAAM,CAAC,QAAQ,CAClB;wBACE,SAAS,IAAI,CAAC,cAAc,WAC1B,IAAI,CAAC,eAAe,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC,SACpC,EAAE;qBACH,EACD,EAAC,WAAW,EAAE,EAAC,WAAW,EAAE,IAAI,EAAC,EAAC,CACnC,CAAC;iBACH;gBACD,MAAM;YACR,KAAK,mBAAmB,CAAC,kBAAkB;gBACzC,IAAI,OAAO,EAAE;oBACX,IAAI,CAAC,MAAM,CAAC,QAAQ,CAClB;wBACE,SAAS,IAAI,CAAC,cAAc,UAC1B,IAAI,CAAC,eAAe,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC,SACpC,EAAE;qBACH,EACD,EAAC,WAAW,EAAE,EAAC,WAAW,EAAE,IAAI,EAAC,EAAC,CACnC,CAAC;iBACH;gBACD,MAAM;YACR,KAAK,mBAAmB,CAAC,aAAa;gBACpC,IAAI,OAAO,EAAE;oBACX,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,EAAE,EAAE;wBACvB,UAAU,EAAE,IAAI,CAAC,WAAW;wBAC5B,WAAW,EAAE,EAAC,WAAW,EAAE,IAAI,EAAC,EAAE,mBAAmB,EAAG,OAAO;qBAChE,CAAC,CAAC;iBACJ;gBACD,MAAM;YACR,KAAK,mBAAmB,CAAC,aAAa;gBACpC,IAAI,OAAO,EAAE;oBACX,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,EAAE,EAAE;wBACvB,UAAU,EAAE,IAAI,CAAC,WAAW;wBAC5B,WAAW,EAAE,EAAC,WAAW,EAAE,IAAI,EAAC,EAAE,mBAAmB,EAAG,OAAO;qBAChE,CAAC,CAAC;iBACJ;gBACD,MAAM;SACT;IACH,CAAC;IAED;;;;;;OAMG;IACH,KAAK,CAAC,oBAAoB,CACxB,WAAmB,EACnB,MAAoB,EACpB,OAAgB,EAChB,MAAkB,IAAI;QAEtB,IAAI,CAAC,eAAe,CAAC,GAAG,CAAC,WAAW,CAAC,CAAC,SAAS,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC;QACjE,IAAI,GAAG,EAAE;YACP,IAAI,CAAC,eAAe,CAAC,GAAG,CAAC,WAAW,CAAC,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC;SACnD;QACD,IAAI,CAAC,eAAe,CAAC,GAAG,CAAC,WAAW,CAAC,CAAC,SAAS,GAAG,MAAM,CAAC,EAAE,CAAC;QAC5D,IAAI;YACF,MAAM,IAAI,CAAC,8BAA8B,CAAC,WAAW,EAAE,MAAM,EAAE,OAAO,CAAC,CAAC;YACxE,iBAAiB;YACjB,MAAM,IAAI,CAAC,6BAA6B,CACtC,WAAW,EACX,MAAM,EACN,OAAO,EACP,GAAG,CACJ,CAAC;SACH;QAAC,OAAO,CAAC,EAAE;YACV,OAAO,CAAC,GAAG,CAAC,kCAAkC,CAAC,EAAE,CAAC,CAAC;SACpD;IACH,CAAC;IAED,KAAK,CAAC,6BAA6B,CACjC,WAAmB,EACnB,MAAoB,EACpB,OAAgB,EAChB,MAAkB,IAAI;QAEtB,MAAM,aAAa,GAAG,MAAM,CAAC,IAAI,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,EAAE,KAAK,GAAG,CAAC,EAAE,CAAC,CAAC;QAC1E,MAAM,CAAC,IAAI,CAAC,KAAK,CAAC,aAAa,CAAC,GAAG,GAAG,CAAC;QACvC,MAAM,EAAC,CAAC,EAAE,CAAC,EAAE,CAAC,EAAC,GAAG,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,UAAU,CAAC,CAAC;QAC7C,MAAM,WAAW,GAAG;YAClB,cAAc,EAAE;gBACd,CAAC,EAAE,CAAC;gBACJ,CAAC,EAAE,CAAC;gBACJ,CAAC,EAAE,CAAC;aACL;YACD,UAAU,EAAE;gBACV,wEAAwE;gBACxE,CAAC,EAAE,CAAC;gBACJ,CAAC,EAAE,GAAG;gBACN,CAAC,EAAE,CAAC;aACL;SACF,CAAC;QACF,IAAI,CAAC,GAAG,CAAC,SAAS,CAAC,YAAY,CAAC,WAAW,EAAE,WAAW,CAAC,CAAC;QAC1D,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC,YAAY,CAAC,WAAW,EAAE,WAAW,CAAC,CAAC;QACpD,IAAI,CAAC,eAAe;aACjB,GAAG,CAAC,WAAW,CAAC;aAChB,WAAW,CAAC,WAAW,CAAC,cAAc,CAAC,CAAC;QAE3C,KAAK,IAAI,KAAK,IAAI,IAAI,CAAC,YAAY,EAAE;YACnC,MAAM,UAAU,GAAG,IAAI,CAAC,eAAe,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC;YACnD,IAAI,CAAC,UAAU,EAAE;gBACf,SAAS;aACV;YACD,IAAI,UAAU,CAAC,SAAS,KAAK,MAAM,CAAC,EAAE,IAAI,KAAK,KAAK,WAAW,EAAE;gBAC/D,IAAI,CAAC,eAAe,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC;gBACnC,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC;gBAC3B,IAAI,CAAC,GAAG,CAAC,SAAS,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC;gBACjC,MAAM;aACP;SACF;IACH,CAAC;IAED;;;;OAIG;IACI,mBAAmB,CAAC,SAAiB;QAI1C,IAAI,KAAK,GAAG,IAAI,CAAC;QACjB,IAAI,OAAO,GAAG,IAAI,CAAC;QACnB,KAAK,IAAI,CAAC,WAAW,EAAE,aAAa,CAAC,IAAI,IAAI,CAAC,eAAe,EAAE;YAC7D,IAAI,aAAa,CAAC,SAAS,KAAK,SAAS,EAAE;gBACzC,KAAK,GAAG,WAAW,CAAC;gBACpB,MAAM,KAAK,GAAG,aAAa,CAAC,UAAU,EAAE,CAAC;gBACzC,IAAI,KAAK,IAAI,IAAI,CAAC,MAAM,IAAI,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,KAAK,CAAC,EAAE;oBACvD,OAAO,GAAG,KAAK,CAAC;iBACjB;aACF;SACF;QACD,OAAO,EAAC,GAAG,EAAE,KAAK,EAAE,KAAK,EAAE,OAAO,EAAC,CAAC;IACtC,CAAC;IAED;;;OAGG;IACH,UAAU;QACR,IAAI,IAAI,CAAC,iBAAiB,EAAE;YAC1B,OAAO,IAAI,CAAC,iBAAiB,CAAC;SAC/B;aAAM;YACL,OAAO,IAAI,CAAC,YAAY,CAAC,MAAM,KAAK,CAAC;gBACnC,CAAC,CAAC,IAAI;gBACN,CAAC,CAAC,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,YAAY,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC;SACrD;IACH,CAAC;IAED;;;;OAIG;IACH,4BAA4B,CAAC,WAAmB;QAC9C,IAAI;YACF,OAAO,IAAI,CAAC,eAAe,CAAC,GAAG,CAAC,WAAW,CAAC,CAAC;SAC9C;QAAC,MAAM;YACN,OAAO,CAAC,GAAG,CAAC,uCAAuC,EAAE,WAAW,CAAC,CAAC;SACnE;QACD,OAAO,IAAI,CAAC;IACd,CAAC;IAED,EAAE;IACF,6DAA6D;IAC7D,EAAE;IACF,KAAK,CAAC,wBAAwB;QAC5B,IAAI,IAAI,CAAC,gBAAgB,IAAI,IAAI,CAAC,iBAAiB,EAAE;YACnD,OAAO,CAAC,GAAG,CAAC,8CAA8C,CAAC,CAAC;YAC5D,OAAO;SACR;QACD,IAAI;YACF,MAAM,IAAI,CAAC,GAAG,CAAC,IAAI,CAAC,MAAM,CAAC,gBAAgB,CAAC,CAAC;SAC9C;QAAC,OAAO,CAAC,EAAE;YACV,OAAO,CAAC,GAAG,CAAC,wBAAwB,EAAE,CAAC,CAAC,CAAC;SAC1C;IACH,CAAC;IAED,0BAA0B;QACxB,IAAI,CAAC,GAAG,CAAC,IAAI,CAAC,MAAM,CAAC,aAAa,CAAC,CAAC;IACtC,CAAC;IAED,mBAAmB;QACjB,IAAI;YACF,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC,OAAO,EAAE,CAAC;SAC1B;QAAC,OAAO,CAAC,EAAE;YACV,OAAO,CAAC,GAAG,CAAC,wBAAwB,EAAE,CAAC,CAAC,CAAC;SAC1C;IACH,CAAC;IAED,KAAK,CAAC,wBAAwB;QAC5B,UAAU,CAAC,KAAK,IAAI,EAAE;YACpB,oFAAoF;YACpF,IAAI,IAAI,CAAC,gBAAgB,IAAI,IAAI,CAAC,iBAAiB,EAAE;gBACnD,OAAO,CAAC,GAAG,CAAC,8CAA8C,CAAC,CAAC;gBAC5D,OAAO;aACR;YACD,IAAI;gBACF,MAAM,IAAI,CAAC,GAAG,CAAC,IAAI,CAAC,MAAM,CAAC,gBAAgB,CAAC,CAAC;aAC9C;YAAC,OAAO,CAAC,EAAE;gBACV,OAAO,CAAC,GAAG,CAAC,wBAAwB,EAAE,CAAC,CAAC,CAAC;aAC1C;QACH,CAAC,EAAE,IAAI,CAAC,CAAC;IACX,CAAC;IAED,qBAAqB;QACnB,MAAM,QAAQ,GAAG,CAAC,IAAI,CAAC,eAAe,CAAC;QACvC,IAAI,CAAC,GAAG,CAAC,YAAY,CAAC,UAAU,CAAC,QAAQ,CAAC,CAAC,IAAI,CAAC,GAAG,EAAE;YACnD,OAAO,CAAC,GAAG,CACT,2BAA2B,QAAQ,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC,UAAU,EAAE,CAC/D,CAAC;QACJ,CAAC,CAAC,CAAC;IACL,CAAC;IAED,4BAA4B;QAC1B,IAAI,IAAI,CAAC,eAAe,EAAE;YACxB,IAAI,CAAC,qBAAqB,EAAE,CAAC;SAC9B;IACH,CAAC;IAED,KAAK,CAAC,kBAAkB,CACtB,SAAiB,EACjB,0BAAkC,IAAI;QAEtC,IAAI,CAAC,IAAI,CAAC,MAAM,EAAE;YAChB,OAAO,CAAC,GAAG,CAAC,0BAA0B,CAAC,CAAC;YACxC,OAAO;SACR;QACD,4BAA4B;QAC5B,kDAAkD;QAClD,IAAI,eAAe,GAAG,IAAI,CAAC,MAAM,CAAC,IAAI,CACpC,CAAC,KAAK,EAAE,EAAE,CAAC,KAAK,CAAC,QAAQ,KAAK,uBAAuB,CACtD,CAAC;QACF,IAAI,CAAC,eAAe,EAAE;YACpB,eAAe,GAAG,IAAI,CAAC,MAAM,CAAC,IAAI,CAChC,CAAC,KAAK,EAAE,EAAE,CAAC,SAAS,CAAC,QAAQ,CAAC,KAAK,CAAC,IAAI,CAAC,IAAI,KAAK,CAAC,IAAI,IAAI,EAAE,CAC9D,CAAC;SACH;QACD,IAAI,CAAC,eAAe,EAAE;YACpB,eAAe,GAAG,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,KAAK,EAAE,EAAE,CAC3C,SAAS,CAAC,QAAQ,CAAC,KAAK,CAAC,EAAE,CAAC,CAC7B,CAAC;SACH;QACD,+BAA+B;QAC/B,IAAI,eAAe,EAAE;YACnB,IAAI,KAAK,GAAG,IAAI,CAAC;YACjB,OAAO,KAAK,EAAE;gBACZ,IAAI;oBACF,MAAM,UAAU,GAAG,MAAM,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC,MAAM,CAC5C,eAAe,CAAC,QAAQ,CACzB,CAAC;oBACF,kDAAkD;oBAClD,KAAK,GAAG,KAAK,CAAC;iBACf;gBAAC,OAAO,KAAK,EAAE;oBACd,OAAO,CAAC,GAAG,CAAC,sBAAsB,EAAE,KAAK,CAAC,CAAC;oBAC3C,MAAM,IAAI,CAAC,GAAG,CAAC,CAAC;iBACjB;aACF;SACF;aAAM;YACL,OAAO,CAAC,IAAI,CAAC,sCAAsC,CAAC,CAAC;SACtD;IACH,CAAC;IAED,KAAK,CAAC,kBAAkB,CACtB,KAAa,EACb,WAAqC,IAAI;QAEzC,IAAI,IAAI,CAAC,eAAe,CAAC,QAAQ,CAAC,KAAK,CAAC,EAAE;YACxC,OAAO,CAAC,GAAG,CAAC,yCAAyC,CAAC,CAAC;YACvD,OAAO;SACR;QACD,oEAAoE;QACpE,UAAU,CAAC,KAAK,IAAI,EAAE;YACpB,IAAI,IAAI,CAAC,gBAAgB,IAAI,IAAI,CAAC,iBAAiB,EAAE;gBACnD,OAAO,CAAC,GAAG,CAAC,mCAAmC,CAAC,CAAC;gBACjD,OAAO;aACR;YACD,IAAI;gBACF,IAAI,CAAC,iBAAiB,GAAG,IAAI,CAAC;gBAC9B,MAAM,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC,MAAM,CAAC,KAAK,EAAE;oBACjC,UAAU,EAAE,oBAAoB;oBAChC,cAAc,EAAE,IAAI;iBACrB,CAAC,CAAC;aACJ;YAAC,OAAO,KAAK,EAAE;gBACd,IAAI,CAAC,iBAAiB,GAAG,KAAK,CAAC;gBAC/B,OAAO,CAAC,GAAG,CAAC,sBAAsB,EAAE,KAAK,CAAC,CAAC;aAC5C;YACD,IAAI,QAAQ,EAAE;gBACZ,MAAM,IAAI,CAAC,GAAG,CAAC,MAAM,CAAC,WAAW,CAAC,QAAQ,EAAE,EAAC,KAAK,EAAE,GAAG,EAAC,CAAC,CAAC,CAAC,8BAA8B;aAC1F;QACH,CAAC,EAAE,IAAI,CAAC,CAAC;IACX,CAAC;IAED,eAAe;QACb,IAAI,IAAI,CAAC,UAAU,EAAE;YACnB,OAAO,IAAI,CAAC,UAAU,CAAC,KAAK,CAAC;SAC9B;QACD,OAAO,IAAI,CAAC;IACd,CAAC;IAED,wBAAwB;QAKtB,IAAI,IAAI,CAAC,UAAU,EAAE;YACnB,OAAO,IAAI,CAAC,UAAU,CAAC;SACxB;QACD,OAAO,IAAI,CAAC;IACd,CAAC;IAED,kBAAkB,CAAC,eAAuB;QACxC,IAAI,CAAC,eAAe,GAAG,eAAe,CAAC;IACzC,CAAC;IAED,kBAAkB;QAChB,OAAO,IAAI,CAAC,eAAe,CAAC;IAC9B,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,QAAQ;QACZ,OAAO,CAAC,GAAG,CAAC,iBAAiB,CAAC,CAAC;QAC/B,IAAI,CAAC,4BAA4B,EAAE,CAAC;QACpC,IAAI,CAAC,MAAM,GAAG,IAAI,CAAC;QACnB,IAAI,CAAC,MAAM,GAAG,IAAI,CAAC;QACnB,IAAI,CAAC,GAAG,GAAG,IAAI,CAAC;QAChB,aAAa,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC;QACjC,IAAI,CAAC,eAAe,GAAG,EAAE,CAAC;QAC1B,IAAI,CAAC,cAAc,GAAG,KAAK,CAAC;QAC5B,uBAAuB;QACvB,IAAI,CAAC,aAAa,CAAC,mBAAmB,CACpC,OAAO,EACP,IAAI,CAAC,uBAAuB,CAC7B,CAAC;QACF,IAAI,CAAC,aAAa,CAAC,mBAAmB,CACpC,aAAa,EACb,IAAI,CAAC,wBAAwB,CAC9B,CAAC;QAEF,wBAAwB;QACxB,IAAI,IAAI,CAAC,uBAAuB,EAAE;YAChC,IAAI,CAAC,uBAAuB,CAAC,mBAAmB,CAC9C,UAAU,EACV,IAAI,CAAC,yBAAyB,CAC/B,CAAC;SACH;QACD,sBAAsB;QACtB,IAAI,CAAC,CAAC,IAAI,CAAC,uBAAuB;YAChC,CAAC,QAAQ,CAAC,QAAQ,CAAC,IAAI,CAAC,OAAO,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC,IAAI,QAAQ,CAAC,QAAQ,CAAC,IAAI,CAAC,OAAO,CAAC,WAAW,CAAC,KAAK,CAAC,CAAC,CAAC,EAAE;YACtG,aAAa,CAAC,IAAI,CAAC,6BAA6B,CAAC,CAAC;SACnD;IACH,CAAC;IAED,KAAK,CAAC,qBAAqB,CAAC,eAAyB;QACnD,IAAI,CAAC,eAAe,GAAG,CAAC,GAAG,eAAe,CAAC,CAAC;QAC5C,IAAI,OAAO,GAAG,CAAC,CAAC;QAChB,MAAM,OAAO,CAAC,GAAG,CACf,eAAe,CAAC,GAAG,CAAC,KAAK,EAAE,KAAK,EAAE,EAAE;YAClC,IAAI;gBACF,MAAM,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC;gBACpC,OAAO,IAAI,CAAC,CAAC;aACd;YAAC,OAAO,KAAK,EAAE;gBACd,OAAO,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC;aACpB;QACH,CAAC,CAAC,CACH,CAAC;QACF,OAAO,CAAC,GAAG,CAAC,iBAAiB,EAAE,OAAO,CAAC,CAAC;IAC1C,CAAC;IAED,EAAE;IACF,qDAAqD;IACrD,EAAE;IACF,KAAK,CAAC,kBAAkB;QACtB,OAAO,IAAI,OAAO,CAAC,KAAK,EAAE,OAAO,EAAE,EAAE;YACnC,IAAI,CAAC,WAAW,CAAC,GAAG,MAAM,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC,aAAa,CAAC,CAAC,CAAC,CAAC;YAE1D,IAAI,IAAI,GAAG,WAAW,CAAC,OAAO,EAAE,CAAC;YAEjC,sBAAsB;YACtB,IAAI,CAAC,YAAY,CAAC,WAAW,CAAC,CAAC;YAC/B,iCAAiC;YACjC;6CACiC;YACjC,IAAI,CAAC,KAAK,EAAE,CAAC;YAEb,MAAM,WAAW,GAAG,WAAW,CAAC,OAAO,EAAE,CAAC;YAC1C,IAAI,CAAC,aAAa,GAAG,WAAW,CAAC,YAAY,CAAC,sBAAsB,CAAC,CAAC;YACtE,WAAW,CAAC,KAAK,EAAE,CAAC;YAEpB,4BAA4B;YAE5B,OAAO,EAAE,CAAC;QACZ,CAAC,CAAC,CAAC;IACL,CAAC;IAED,KAAK,CAAC,WAAW,CACf,GAAc,EACd,IAA6B;QAE7B,OAAO,IAAI,OAAO,CAAC,KAAK,EAAE,OAAO,EAAE,EAAE;YACnC,MAAM,CAAC,WAAW,CAAC,GAAG,MAAM,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC,aAAa,CAAC,CAAC,CAAC,CAAC;YAE5D,kEAAkE;YAClE,iBAAiB;YACjB,iBAAiB;YACjB,QAAQ;YACR,IAAI,wBAAwB,GAAG,KAAK,CAAC;YACrC,IAAI,gBAAgB,GAAG,KAAK,CAAC;YAC7B,IAAI,6BAA6B,GAAG,KAAK,CAAC;YAC1C,IAAI,gBAAgB,GAAG,KAAK,CAAC;YAE7B;;eAEG;YACH,IAAI,GAAG,CAAC,MAAM,KAAK,iBAAiB,EAAE;gBACpC,wBAAwB,GAAG,IAAI,CAAC;aACjC;YACD,IAAI,GAAG,CAAC,MAAM,KAAK,iBAAiB,EAAE;gBACpC,gBAAgB,GAAG,IAAI,CAAC;aACzB;YACD,IAAI,GAAG,CAAC,MAAM,KAAK,OAAO,EAAE;gBAC1B,6BAA6B,GAAG,IAAI,CAAC;aACtC;YACD,IAAG,GAAG,CAAC,MAAM,KAAK,SAAS,EAAE;gBAC3B,gBAAgB,GAAG,IAAI,CAAC;aACzB;YAED,MAAM,SAAS,GAAG,WAAW,CAAC,OAAO,EAAE,CAAC;YACxC,IAAI,SAAS,GAAG,IAAI,CAAC;YACrB,MAAM,OAAO,GAAG;gBACd,GAAG,EAAE,6BAA6B,GAAG,CAAC,MAAM,GAC1C,GAAG,CAAC,MAAM,CAAC,OAAO,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC,GAAG,GAAG,GAAG,CAAC,MAAM,CAAC,CAAC,CAAC,GAAG,CAAC,MAC1D,EAAE;gBACF,sGAAsG;gBACtG,aAAa,EAAE,EAAC,GAAG,EAAE,CAAC,EAAE,GAAG,EAAE,CAAC,EAAE,GAAG,EAAE,CAAC,EAAC;gBACvC,sGAAsG;gBACtG,aAAa,EAAE,EAAC,GAAG,EAAE,CAAC,EAAE,GAAG,EAAE,CAAC,EAAE,GAAG,EAAE,CAAC,EAAC;gBACvC,OAAO,EAAE,IAAI;gBACb,eAAe,EAAE,IAAI;aACtB,CAAC;YAGF,QAAQ,GAAG,CAAC,MAAM,EAAE;gBAClB,KAAK,MAAM,CAAC;gBACZ,KAAK,KAAK;oBACR,SAAS,GAAG,SAAS,CAAC,YAAY,CAAC,cAAc,EAAE,OAAO,CAAC,CAAC;oBAC5D,MAAM;gBACR,KAAK,MAAM,CAAC;gBACZ,KAAK,KAAK;oBACR,SAAS,GAAG,SAAS,CAAC,YAAY,CAAC,cAAc,EAAE,OAAO,CAAC,CAAC;oBAC5D,MAAM;gBACR,KAAK,OAAO,CAAC;gBACb,KAAK,MAAM;oBACT,SAAS,GAAG,SAAS,CAAC,YAAY,CAAC,eAAe,EAAE,OAAO,CAAC,CAAC;oBAC7D,MAAM;gBACR,KAAK,MAAM,CAAC;gBACZ,KAAK,KAAK;oBACR,SAAS,GAAG,SAAS,CAAC,YAAY,CAAC,eAAe,EAAE,OAAO,CAAC,CAAC;oBAC7D,MAAM;gBACR,KAAK,MAAM,CAAC;gBACZ,KAAK,KAAK;oBACR,SAAS,GAAG,SAAS,CAAC,YAAY,CAAC,cAAc,EAAE,OAAO,CAAC,CAAC;oBAC5D,MAAM;gBAER;oBACE,OAAO,CAAC,GAAG,CAAC,sBAAsB,CAAC,CAAC;oBACpC,MAAM;aACT;YAED,mDAAmD;YACnD;;;;;;4CAMgC;YAChC,8CAA8C;YAC9C,qBAAqB;YAErB,6BAA6B;YAC7B,IAAI,IAAI,CAAC,iBAAiB,EAAE;gBAC1B,MAAM,UAAU,GAAG,WAAW,CAAC,OAAO,EAAE,CAAC;gBACzC,UAAU,CAAC,YAAY,CAAC,iBAAiB,EAAE;oBACzC,SAAS,EAAE,CAAC;oBACZ,KAAK,EAAE,EAAC,CAAC,EAAE,GAAG,EAAE,CAAC,EAAE,GAAG,EAAE,CAAC,EAAE,GAAG,EAAC;iBAChC,CAAC,CAAC;gBACH,IAAI,CAAC,iBAAiB,GAAG,KAAK,CAAC;aAChC;YAED,SAAS,CAAC,KAAK,CAAC,QAAQ,CAAC,GAAG,CAC1B,GAAG,CAAC,QAAQ,CAAC,CAAC,EACd,GAAG,CAAC,QAAQ,CAAC,CAAC,EACd,GAAG,CAAC,QAAQ,CAAC,CAAC,CACf,CAAC;YAEF,SAAS,CAAC,KAAK,CAAC,QAAQ,CAAC,GAAG,CAC1B,GAAG,CAAC,QAAQ,CAAC,CAAC,EACd,GAAG,CAAC,QAAQ,CAAC,CAAC,EACd,GAAG,CAAC,QAAQ,CAAC,CAAC,CACf,CAAC;YACF,SAAS,CAAC,KAAK,CAAC,KAAK,CAAC,GAAG,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC,EAAE,GAAG,CAAC,KAAK,CAAC,CAAC,EAAE,GAAG,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;YACjE,IAAG,gBAAgB,EACnB;gBACE;;mBAEG;gBACH,IAAI,CAAC,cAAc,GAAC,SAAS,CAAC;gBAC9B,IAAI,CAAC,oBAAoB,EAAE,CAAC;aAC7B;YAED,2DAA2D;YAC3D,sFAAsF;YACtF,IAAI,IAAI,IAAI,CAAC,gBAAgB,EAAE;gBAC7B,IAAI,CAAC,4BAA4B,CAC/B,SAAS,EACT,WAAW,EACX,IAAI,EACJ,IAAI,EACJ,IAAI,CACL,CAAC,KAAK,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC;aAChC;YAED,IAAI,IAAI,CAAC,YAAY,IAAI,OAAO,IAAI,CAAC,YAAY,CAAC,EAAE,KAAK,QAAQ,EAAE;gBACjE,8EAA8E;gBAC9E,SAAS,CAAC,KAAK,CAAC,IAAI,GAAG,IAAI,CAAC,YAAY,CAAC,EAAE,CAAC;aAC7C;iBAAM,IAAI,GAAG,CAAC,EAAE,EAAE;gBACjB,SAAS,CAAC,KAAK,CAAC,IAAI,GAAG,GAAG,CAAC,EAAE,CAAC;aAC/B;YACD,IAAI,CAAC,YAAY,GAAG,SAAS,CAAC,KAAK,CAAC;YAEpC,uCAAuC;YACvC,iCAAiC;YACjC,yBAAyB;YACzB,IAAI,CAAC,oBAAoB,CAAC,GAAG,CAAC,SAAS,CAAC,KAAK,CAAC,IAAI,EAAE,SAAS,CAAC,CAAC;YAC/D,IAAI,CAAC,yBAAyB,CAAC,GAAG,CAAC,SAAS,CAAC,KAAK,CAAC,IAAI,EAAE,WAAW,CAAC,CAAC;YAEtE;;;;;;;;;;;;;;8BAckB;YAElB,IAAI,wBAAwB,EAAE;gBAE5B,MAAM,QAAQ,GAAG,SAAS,CAAC,YAAY,CAAC,mBAAmB,EACzD;oBACE,WAAW,EAAE,GAAG;oBAChB,UAAU,EAAE,EAAE;oBACd,eAAe,EAAE,EAAE;oBACnB,QAAQ,EAAE,kBAAkB;oBAC5B,eAAe,EAAE;wBACf,GAAG,EAAE,GAAG;wBACR,GAAG,EAAE,IAAI;wBACT,GAAG,EAAE,CAAC;qBACP;oBACD,eAAe,EAAE;wBACf,GAAG,EAAE,CAAC,EAAE;wBACR,GAAG,EAAE,CAAC,EAAE;wBACR,GAAG,EAAE,CAAC;qBACP;oBACD,OAAO,EAAE,KAAK;oBACd,WAAW,EAAE,CAAC;oBACd,UAAU,EAAE,CAAC,EAAE;iBAChB,CAAC,CAAC;gBACL,MAAM,WAAW,GAAG,WAAW,CAAC,OAAO,CAAC;oBACtC,EAAE,EAAE,gBAAgB;oBACpB,IAAI,EAAE,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC,QAAQ,CAAC,MAAM;oBACpC,IAAI,EAAE,SAAS;oBACf,SAAS,EAAE,QAAQ;oBACnB,QAAQ,EAAE,YAAY;iBACvB,CAAC,CAAC;gBAEH,IAAI,CAAC,sBAAsB,GAAG,QAAQ,CAAC;gBACvC,IAAI,CAAC,GAAG,CAAC,WAAW,EAAE;oBACpB,UAAU,CAAC,GAAG,EAAE;wBACd,QAAQ,CAAC,iBAAiB,EAAE,CAAC;oBAC/B,CAAC,EAAE,IAAI,CAAC,CAAC;iBACV;aACF;YAED,IAAI,gBAAgB,EAAE;gBACpB,gFAAgF;gBAChF,iCAAiC;gBACjC,gCAAgC;gBAChC,eAAe;gBACf,MAAM,EAAE,GAAG,IAAI,cAAc,EAAE,CAAC;gBAChC,EAAE,CAAC,MAAM,EAAE,CAAC;gBACZ,MAAM,KAAK,GAAG,IAAI,aAAa,EAAE,CAAC;gBAClC,KAAK,CAAC,OAAO,GAAG;oBACd,UAAU,EAAE,IAAI,QAAQ,EAAE;oBAC1B,QAAQ,EAAE,IAAI,QAAQ,EAAE;iBACzB,CAAC;gBACF,MAAM,YAAY,GAAG,EAAE,CAAC,OAAO,CAAC,OAAO,CAAC;gBACxC,KAAK,CAAC,YAAY,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC;gBACtC,KAAK,CAAC,MAAM,CAAC,SAAS,EAAE,YAAY,CAAC,CAAC;gBACtC,MAAM,EAAE,GAAG,IAAI,cAAc,EAAE,CAAC;gBAChC,EAAE,CAAC,YAAY,CAAC,SAAS,EAAE,KAAK,EAAE,EAAE,CAAC,CAAC;gBACtC,EAAE,CAAC,YAAY,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC;gBACnC,EAAE,CAAC,MAAM,CAAC,SAAS,EAAE,KAAK,EAAE,YAAY,CAAC,CAAC;gBAC1C,EAAE,CAAC,8BAA8B,CAAC,EAAE,CAAC,CAAC;gBACtC,UAAU,CAAC,GAAG,EAAE;oBACd,EAAE,CAAC,MAAM,CAAC,YAAY,GAAG,QAAQ,CAAC;oBAClC,EAAE,CAAC,eAAe,EAAE,CAAC;gBACvB,CAAC,EAAE,IAAI,CAAC,CAAC;aACV;YAED,IAAI,6BAA6B,EAAE;gBACjC,gFAAgF;gBAChF,MAAM,EAAE,GAAG,IAAI,QAAQ,EAAE,CAAC;gBAC1B,EAAE,CAAC,YAAY,CAAC,SAAS,CAAC,CAAC;gBAC3B,EAAE,CAAC,MAAM,CAAC,SAAS,CAAC,CAAC;gBACrB,UAAU,CAAC,GAAG,EAAE;oBACd,EAAE,CAAC,MAAM,CAAC,YAAY,GAAG,QAAQ,CAAC;oBAClC,EAAE,CAAC,eAAe,EAAE,CAAC;gBACvB,CAAC,EAAE,IAAI,CAAC,CAAC;aACV;YAED,WAAW,CAAC,KAAK,EAAE,CAAC;YACpB,OAAO,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC;QAC7B,CAAC,CAAC,CAAC;IACL,CAAC;IAED,sBAAsB,CAAC,QAAa;QAClC,IAAI,GAAG,GAAG,IAAI,CAAC,oBAAoB,CAAC,GAAG,CAAC,QAAQ,CAAC,CAAC;QAClD,GAAG,CAAC,KAAK,CAAC,OAAO,GAAG,CAAC,GAAG,CAAC,KAAK,CAAC,OAAO,CAAC;IACzC,CAAC;IAED,eAAe,CAAC,QAAa;QAC3B,IAAI,GAAG,GAAG,IAAI,CAAC,oBAAoB,CAAC,GAAG,CAAC,QAAQ,CAAC,CAAC;QAClD,OAAO,GAAG,CAAC,KAAK,CAAC,OAAO,CAAC;IAC3B,CAAC;IAED,KAAK,CAAC,qBAAqB,CAAC,QAAa;QACvC,IAAI,GAAG,GAAG,IAAI,CAAC,oBAAoB,CAAC,GAAG,CAAC,QAAQ,CAAC,CAAC;QAElD,2BAA2B;QAC3B,MAAM,SAAS,GAAG;YAChB,UAAU,EAAE,IAAI,CAAC,SAAS,CAAC,GAAG,CAAC,KAAK,CAAC,QAAQ,CAAC;YAC9C,IAAI,EAAE,OAAO,CAAC,QAAQ;YACtB,SAAS,EAAE,QAAQ,EAAE,4BAA4B;SAC3C,CAAC;QACT,MAAM,QAAQ,GAAG,EAAC,EAAE,EAAE,QAAQ,EAAiB,CAAC;QAEhD,IAAG;YACD,MAAM,IAAI,CAAC,sBAAsB,CAAC,OAAO,CAAC,QAAQ,EAAE,QAAQ,EAAE,SAAS,CAAC,CAAC;SAC1E;QAAA,OAAM,GAAQ,EAAC,GAAE;QAElB,iEAAiE;QACjE,wDAAwD;QACxD,oEAAoE;QAEpE,IAAI,MAAM,GAAG,IAAI,CAAC,mBAAmB,CAAC,QAAQ,CAAC,CAAC;QAEhD,MAAM,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC;QAC/B,IAAI,CAAC,mBAAmB,EAAE,CAAC;IAC7B,CAAC;IAED,0BAA0B,CAAC,IAAY;QACrC,OAAO,IAAI,CAAC,yBAAyB,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;IAClD,CAAC;IAED,KAAK,CAAC,oBAAoB;QACxB,IAAG,IAAI,CAAC,cAAc,EAAC;YACrB,IAAI,CAAC,cAAc,CAAC,KAAK,CAAC,QAAQ,CAAC,GAAG,CAAC,IAAI,CAAC,UAAU,CAAC,QAAQ,CAAC,CAAC,EAAC,IAAI,CAAC,UAAU,CAAC,QAAQ,CAAC,CAAC,EAAC,IAAI,CAAC,UAAU,CAAC,QAAQ,CAAC,CAAC,CAAC,CAAA;SACzH;IACH,CAAC;IAED,KAAK,CAAC,4BAA4B,CAChC,SAAc,EACd,WAAgB,EAChB,IAAsC,EACtC,OAAgB,EAChB,WAAoB;QAEpB,0DAA0D;QAC1D,IAAI,IAAI,GAAG,IAAI,CAAC;QAChB,IAAI,GAAG,WAAW,CAAC,OAAO,EAAE,CAAC;QAC7B,IAAI,CAAC,IAAI,EAAE;YACT,MAAM,CAAC,WAAW,CAAC,GAAG,MAAM,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC,aAAa,CAAC,CAAC,CAAC,CAAC;YAC5D,IAAI,GAAG,WAAW,CAAC,OAAO,EAAE,CAAC;SAC9B;QAED,MAAM,SAAS,GAAG,IAAI,CAAC,YAAY,CAAC,sBAAsB,CAAC,CAAC;QAC5D,IAAI,CAAC,KAAK,EAAE,CAAC;QACb,EAAE;QACF,+FAA+F;QAC/F,SAAS,CAAC,iBAAiB,CAAC,OAAO,GAAG,OAAO,CAAC;QAC9C,EAAE;QACF,+CAA+C;QAC/C,SAAS,CAAC,MAAM,CAAC,SAAS,GAAG,SAAS,CAAC;QACvC,EAAE;QACF,qDAAqD;QACrD,SAAS,CAAC,MAAM,CAAC,IAAI,GAAG,IAAI,CAAC;QAC7B,SAAS,CAAC,KAAK,CAAC,QAAQ,GAAG,SAAS,CAAC,CAAC,gCAAgC;QACtE,IAAG,WAAW,EAAC,EAAE,6CAA6C;YAC5D,IAAI,CAAC,IAAI,CAAC,YAAY,IAAI,CAAC,IAAI,CAAC,YAAY,CAAC,QAAQ,EAAE;gBACrD,IAAI;oBACF,SAAS,CAAC,KAAK,CAAC,IAAI,GAAG,IAAI,CAAC,YAAY,CAAC,IAAI,IAAI,IAAI,CAAC,YAAY,CAAC,EAAE,CAAC;iBACvE;gBAAC,OAAO,CAAC,EAAE;oBACV,OAAO,CAAC,GAAG,CAAC,sDAAsD,CAAC,CAAC;iBACrE;gBACD,IAAI,CAAC,YAAY,GAAG,SAAS,CAAC,KAAK,CAAC;aACrC;SACF;aAAI,EAAC,qEAAqE;YACzE,OAAO,CAAC,GAAG,CAAC,kBAAkB,CAAC,CAAC;YAChC,OAAO,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC;YACvB,OAAO,CAAC,GAAG,CAAC,SAAS,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;YAClC,IAAI,CAAC,YAAY,GAAG,SAAS,CAAC,KAAK,CAAC;SACrC;QAED,OAAO,SAAS,CAAC;IACnB,CAAC;IAED,yBAAyB;QACvB,IAAI,CAAC,YAAY,CAAC,QAAQ,CAAC,iBAAiB,CAAC,OAAO,GAAG,KAAK,CAAC;QAC7D,IAAI,CAAC,YAAY,CAAC,QAAQ,CAAC,iBAAiB,CAAC,OAAO,EAAE,CAAC;QACvD,IAAI,CAAC,YAAY,CAAC,QAAQ,GAAG,IAAI,CAAC;IACpC,CAAC;IAED,iBAAiB;QACf,IAAI,CAAC,sBAAsB,CAAC,iBAAiB,EAAE,CAAC;IAClD,CAAC;;8GAl9DU,iBAAiB,kBAqJlB,QAAQ;kHArJP,iBAAiB,cAFhB,MAAM;2FAEP,iBAAiB;kBAH7B,UAAU;mBAAC;oBACV,UAAU,EAAE,MAAM;iBACnB;;0BAsJI,MAAM;2BAAC,QAAQ","sourcesContent":["import {Inject, Injectable, NgZone} from '@angular/core';\nimport {ActivatedRoute, Router} from '@angular/router';\nimport {Subject} from 'rxjs';\n//import * as serialijse from \"serialijse\";\nimport {\n  getDistanceBetweenTwoPoints,\n  poiTypeToString,\n  wait,\n} from '../helpers.service';\nimport {MattertagData} from '../mattertagData';\nimport {\n  DbObjectType,\n  FeatureType,\n  IObject3D,\n  MattertagActionMode,\n  POI,\n  PoiType,\n  SpModule,\n  TagAction,\n  ViewerInteractions,\n} from '../types.service';\nimport {Object3D} from 'three';\nimport {BaseTagService} from './tag.service';\nimport {BaseVisibilityService} from './baseVisibility.service';\nimport {Config} from '../config';\nimport {CameraMode} from '../types.service';\nimport {ISceneNode, ComponentOutput} from \"../matterport-extensions/scene-component/SceneComponent\";\nimport {SecurityCamera} from '../matterport-extensions/security-camera/SecurityCamera';\nimport {NestThermostat} from '../matterport-extensions/nest-thermostat/NestThermostat';\nimport {PlaneRenderer} from '../matterport-extensions/nest-thermostat/PlaneRenderer';\nimport {CanvasRenderer} from \"../matterport-extensions/nest-thermostat/CanvasRenderer\";\nimport {VideoRenderer} from \"../matterport-extensions/video-renderer/VideoRenderer\";\nimport {HlsLoader} from \"../matterport-extensions/hsl-loader/HlsLoader\";\nimport {TvPlayer} from \"../matterport-extensions/tv-player/TvPlayer\";\n\n// import {CanvasImage} from \"../matterport-extensions/nest-thermostat/CanvasImage\";\n\n/**\n * Custom Threejs Object 3D interface\n */\ninterface ThreeJSObject3D {\n  id: string;\n  uuid?: string;\n  position?: { x: number, y: number, z: number }\n  rotation?: { x: number, y: number, z: number }\n  scale?: { x: number, y: number, z: number }\n}\n\ndeclare global {\n  interface Window {\n    MP_SDK: { connect: Function };\n  }\n}\n\n@Injectable({\n  providedIn: 'root',\n})\nexport class MatterportService {\n\n  private slots: any[] = [];//SlotNode[] = [];\n\n  private nodes: ISceneNode[] = [];\n\n  public sdk: any;\n\n  private container: any;\n\n  // Position camera\n  private poseMatterport: any;\n\n  public poseCamera: { rotation: any; position: any; sweep: any };\n  public lastCameraPosition: any = {x:0.0,y:0.0,z:0.0};\n\n  private azimuthalCrown:any;\n  // Pointer trick\n  private pointerButton: any;\n\n  // Display and get current position of cursor (for Admins only)\n  private getCursorPositionButton: any;\n  private cursorPositionButtonDisplayed = false;\n  private intervalCursorPointerPosition: any;\n  private textDisplayCursorPositionPanel: any;\n\n  private oldPoseMatterportPosition: any;\n  // Measure mode\n  private isMeasureModeOn = false;\n\n  private interactionMode: number = ViewerInteractions.DEFAULT;\n\n  // List of created Mattertag IDs in the current viewer session\n  private mattertagIDs: Array<string> = [];\n\n  // Dictionnary of MattertagID and its data (mattertagData)\n  private dictionnaryTags: Map<string, MattertagData> = new Map();\n\n  private dictionnaryObjects3D: Map<string, any> = new Map();\n\n  private dictionnarySceneObjects3D: Map<string, any> = new Map();\n\n  public threeJSScene: any; // global root scene (not used now!)\n\n  private lastMeasure = [];\n\n  private distancesLastMeasure: number[] = [];\n\n  private resolution = {\n    width: 500,\n    height: 600,\n  };\n\n  private visibility = {\n    mattertags: false,\n    sweeps: true,\n  };\n\n  public tagsAttachments: Object = {};\n\n  private lastScreenshotUri: any;\n\n  public sweeps: Array<string> | null;\n\n  public currentSweep: Subject<string> = new Subject();\n\n  //camera position with rotation\n  public currentCameraPose: Subject<any> = new Subject();\n\n  private floors?: Array<{ id: string; name: string; sequence: number }> | null;\n\n  private _currentSpaceID: string;\n\n  private lastObject3D: any; // Three JS object\n\n  private currentFloor;\n\n  public get currentSpaceID(): string {\n    return this._currentSpaceID;\n  }\n\n  public set currentSpaceID(value: string) {\n    this._currentSpaceID = value;\n  }\n\n  timerPointer: any;\n\n  forbiddenSweeps: string[] = [];\n\n  public tagAction: Subject<{ action: string; data: string }> = new Subject();\n\n  public mattertagToFollow: string | null;\n\n  tagService: BaseTagService;\n\n  config: Config;\n\n  public inTransitionMode = false;\n\n  public inTransitionSweep = false;\n\n  private noLightForObjects = true;\n\n  public currentCameraMode: CameraMode = CameraMode.OUTSIDE;\n\n  public onCameraModeChanged = new Subject<CameraMode>();\n\n  public onGoToTag = new Subject<string>();\n\n  tagMessengerOn = false;\n\n  SPModule: SpModule;\n\n  public objectControl: any;\n\n  private securityCameraAnimator: any;\n\n  /**\n   * Actions on left click when positioning mattertag in visit\n   */\n  pointerLeftClickHandler = () => {\n    if (this.mattertagToFollow) {\n      const mattertagData = this.dictionnaryTags.get(this.mattertagToFollow);\n      mattertagData.setPosition({...this.poseMatterport.position}); // copy!! not the reference\n      mattertagData.setNormal({...this.poseMatterport.normal}); // copy!! not the reference\n      this.dictionnaryTags.set(this.mattertagToFollow, mattertagData);\n      this.updateMatterTagContentForTagID(this.mattertagToFollow);\n    }\n    this.onValidatedMattertag();\n  };\n\n  pointerRightClickHandler = (e) => {\n    e.preventDefault();\n    this.cancelFollowingCursor();\n    alert('action cancelled');\n  };\n\n  pointerMiddleClickHandler = (e) => {\n    this.textDisplayCursorPositionPanel.innerHTML = `position:\n       ${this.pointToString(this.poseMatterport.position)}\\n\n       normal:\n       ${this.pointToString(this.poseMatterport.normal)}\\n\n       floorId:\n       ${this.poseMatterport.floorId}`;\n    // this.textDisplayCursorPositionPanel.style.display = 'visible';\n    this.getCursorPositionButton.style.display = 'none';\n  }\n\n  constructor(\n    @Inject('config') config: Config,\n    private router: Router,\n    private activeRoute: ActivatedRoute,\n    private visibilityService: BaseVisibilityService,\n    private ngZone: NgZone\n  ) {\n    this.config = config;\n\n    // TODO: only for dev!\n    if (!!this.getCursorPositionButton &&\n      (document.location.href.indexOf('dev') !== -1 || document.location.href.indexOf('localhost') !== -1)) {\n      this.intervalCursorPointerPosition = setInterval(() => {\n        if (!this.poseMatterport) {\n          return;\n        }\n\n        const nextShow = this.poseMatterport.time + 1000;\n        if (new Date().getTime() > nextShow) {\n          if (this.cursorPositionButtonDisplayed) {\n            return;\n          }\n\n          const size = {\n            w: this.container.clientWidth,\n            h: this.container.clientHeight,\n          };\n          const coord = this.sdk.Conversion.worldToScreen(this.poseMatterport.position, this.poseCamera, size);\n          this.getCursorPositionButton.style.left = `${coord.x - 25}px`;\n          this.getCursorPositionButton.style.top = `${coord.y - 22}px`;\n          this.getCursorPositionButton.style.display = 'block';\n          // this.textDisplayCursorPositionPanel.style.display = 'none';\n          this.cursorPositionButtonDisplayed = true;\n        }\n      }, 500);\n    }\n  }\n\n  /**\n   * Initializes Matterport and all listeners/data\n   * @param tagService BaseTagService (to inject html)\n   * @param module SpModule (Museum, Immo) to subscribe only to needed functionnnalities\n   * @returns boolean\n   */\n  async initSdk(\n    tagService: BaseTagService,\n    module: SpModule = SpModule.IMMO\n  ): Promise<boolean> {\n    if (this.sdk) {\n      // clean if sdk is running already\n      await this.action_delete_all_mattertags();\n    }\n    this.tagService = tagService;\n    this.SPModule = module;\n    return new Promise((resolve, reject) => {\n      // Retrieve DOM elements\n      this.pointerButton = document.querySelector(\n        '#viewer-pointer-trick'\n      ) as HTMLElement;\n      this.container = document.querySelector('#viewer-module') as HTMLElement;\n      const iframe = document.querySelector(\n        '#viewer-module'\n      ) as HTMLIFrameElement;\n      if (!iframe) {\n        return;\n      }\n\n      // Add listeners\n      if (this.pointerButton) {\n        this.pointerButton.addEventListener(\n          'click',\n          this.pointerLeftClickHandler\n        );\n        // cancel on right click\n        this.pointerButton.addEventListener(\n          'contextmenu',\n          this.pointerRightClickHandler\n        );\n      }\n      // Retrieve DOM elements\n      this.getCursorPositionButton = document.querySelector(\n        '#button'\n      ) as HTMLElement;\n      this.textDisplayCursorPositionPanel = document.querySelector(\n        '#text'\n      ) as HTMLElement;\n      if (this.getCursorPositionButton) {\n        // get position on Matterport \"model\" on middle click\n        this.getCursorPositionButton.addEventListener(\n          'click',\n          this.pointerMiddleClickHandler\n        );\n      }\n\n      // Load SDK\n      console.log('Loading Matterport SDK');\n      const showcaseWindow = iframe.contentWindow;\n      iframe.addEventListener(\n        'load',\n        async function () {\n          try {\n            this.sdk = await showcaseWindow.MP_SDK.connect(\n              showcaseWindow,\n              'qn9wsasuy5h2fzrbrn1nzr0id',\n              '3.5'\n            );\n          } catch (e) {\n            console.error(e);\n            return;\n          }\n\n          // Load Mattertag icons and custom subscriptions\n          switch (module) {\n            case SpModule.IMMO:\n              this.sdk.Asset.registerTexture(\n                'icon-ticket',\n                this.config.my_config.icon_ticket\n              );\n              this.sdk.Asset.registerTexture(\n                'icon-equipment',\n                this.config.my_config.icon_equipment\n              );\n              this.sdk.Asset.registerTexture(\n                'icon-measure',\n                this.config.my_config.icon_measure\n              );\n              this.sdk.Asset.registerTexture(\n                'icon-data',\n                this.config.my_config.icon_data\n              );\n              this.sdk.Asset.registerTexture(\n                'icon-object3d',\n                this.config.my_config.icon_object3d\n              );\n\n              this.sdk.Measurements.data.subscribe({\n                onAdded: function (\n                  index: any,\n                  item: { points: any },\n                  collection: any\n                ) {\n                  // console.log(\n                  //     \"item added to the collection\",\n                  //     index,\n                  //     item,\n                  // );\n                  // this.measurements[index] = item;\n                  this.lastMeasure = item.points;\n                }.bind(this),\n                onCollectionUpdated: function (collection: any) {\n                  // console.log('the entire up-to-date collection', collection);\n                  this.getDistanceForLastMeasurement();\n                }.bind(this),\n              });\n              this.sdk.Measurements.mode.subscribe(\n                function (measurementModeState: { active: any }) {\n                  // measurement mode state has changed\n                  this.isMeasureModeOn = measurementModeState.active;\n                  // console.log('Is measurement mode currently active? ', measurementModeState.active);\n                }.bind(this)\n              );\n              break;\n            case SpModule.MUSEUM:\n              this.sdk.Asset.registerTexture(\n                'icon-data',\n                this.config.my_config.icon_data\n              );\n              break;\n            case SpModule.HOTEL:\n              this.sdk.Asset.registerTexture(\n                'icon_room_available',\n                this.config.my_config.icon_room_available\n              );\n              this.sdk.Asset.registerTexture(\n                'icon_room_unavailable',\n                this.config.my_config.icon_room_unavailable\n              );\n              this.sdk.Asset.registerTexture(\n                'icon_room_chosen',\n                this.config.my_config.icon_room_chosen\n              );\n              this.sdk.Asset.registerTexture(\n                'icon_room_viewed',\n                this.config.my_config.icon_room_viewed\n              );\n              break;\n            default:\n              break;\n          }\n\n          this.sdk.Asset.registerTexture(\n            'icon-position',\n            this.config.my_config.icon_position\n          );\n\n          // Current Room (used for getting bounding box and eventually lazy load mattertag or object3D inside current Room)\n          this.sdk.Room.current.subscribe((currentRoom) => {\n            console.log(`hello current Room ${JSON.stringify(currentRoom)}`);\n          });\n\n          // 3D position's pointer\n          this.sdk.Pointer.intersection.subscribe(\n            function (intersection: any) {\n              this.poseMatterport = intersection;\n              this.poseMatterport.time = new Date().getTime();\n              if (!this.oldPoseMatterportPosition) {\n                this.oldPoseMatterportPosition = {\n                  x: 0,\n                  y: 0,\n                  z: 0,\n                };\n              }\n\n              if (\n                this.interactionMode !== ViewerInteractions.DEFAULT &&\n                this.mattertagToFollow\n              ) {\n                // follow the pointer and changes the position of the last tag\n                // (we are about to validate, but it exists in sdk already)\n                this.enable_following_tag(this.mattertagToFollow);\n              }\n              if (!!this.getCursorPositionButton && !!this.getCursorPositionButton.style &&\n              (document.URL.indexOf('https://dev.smarterplan.io') !== -1 || document.location.href.indexOf('localhost') !== -1)) {\n                this.getCursorPositionButton.style.display = 'none';\n                this.cursorPositionButtonDisplayed = false;\n              }\n            }.bind(this)\n          );\n\n          //Camera mode\n          this.sdk.Mode.current.subscribe((mode) => {\n            this.inTransitionSweep = false;\n            switch (mode) {\n              case 'mode.dollhouse':\n                this.currentCameraMode = CameraMode.DOLLHOUSE;\n                break;\n              case 'mode.floorplan':\n                this.currentCameraMode = CameraMode.FLOORPLAN;\n                break;\n              case 'mode.inside':\n                this.currentCameraMode = CameraMode.INSIDE;\n                break;\n              case 'mode.transitioning':\n                this.currentCameraMode = CameraMode.TRANSITIONING;\n                break;\n              default:\n                this.currentCameraMode = CameraMode.OUTSIDE;\n            }\n            this.onCameraModeChanged.next(this.currentCameraMode);\n          });\n\n          // Camera's viewpoint\n          this.sdk.Camera.pose.subscribe(\n            function subscr(pose: any) {\n              this.poseCamera = pose;\n\n              this.currentCameraPose.next(pose);\n              // console.log('Current position is ', pose.position);\n              // console.log('Rotation angle is ', pose.rotation);\n              // console.log(\"Sweep UUID is\", pose.sweep);\n            }.bind(this)\n          );\n\n          // subscribe to sweeps\n          this.sdk.Sweep.data.subscribe({\n            onCollectionUpdated: function subscr(collection: {}) {\n              // console.log(\"Sweep collection updated\");\n              this.sweeps = Object.keys(collection);\n            }.bind(this),\n          });\n\n          // subscribe to current sweep\n          this.sdk.Sweep.current.subscribe(\n            function subscr(currentSweep: { sid: string }) {\n              // Change to the current sweep has occurred.\n              if (currentSweep.sid === '') {\n                // console.log('Not currently stationed at a sweep position');\n              } else {\n                // console.log(\"emmiting sweep\", currentSweep.sid);\n                this.currentSweep.next(currentSweep.sid);\n              }\n            }.bind(this)\n          );\n\n          // Subscribe to Floor data\n          this.sdk.Floor.data.subscribe({\n            onCollectionUpdated: function upd(this: any, collection: any) {\n              this.floors = Object.values(collection);\n            }.bind(this),\n          });\n\n          // on tag click open details page\n          this.sdk.on('tag.click', (sid: string) => {\n            // get object of this tag\n            try {\n              const mattertagData = this.dictionnaryTags.get(sid);\n              if (!mattertagData){\n                console.log('no mattertagData to display');\n                return;\n              }\n              const url = tagService.getUrlForSeeDetails(\n                mattertagData.getObject(),\n                mattertagData.getType()\n              );\n              if (url !== '') {\n                this.visibilityService.detailShowing.next(true);\n                this.ngZone.run(() => {\n                  this.router.navigate([url]);\n                });\n              }\n            } catch {\n              console.log('Cannot show details for tag');\n            }\n          });\n\n          // Pointer trick\n          // Create a div that will appear when the cursor is still\n          // It will intercept the click of the mouse before it trigerring the Matterport's tour navigation\n          this.timerPointer = setInterval(\n            this.updatePointerTrick.bind(this),\n            50\n          );\n\n          /**\n           * Transitions\n           */\n          this.sdk.on(\n            'viewmode.changestart',\n            function (to, from) {\n              // console.log('Starting to move to ' + to + ' from ' + from);\n              this.inTransitionMode = true;\n            }.bind(this)\n          );\n\n          this.sdk.on(\n            'viewmode.changeend',\n            function (oldMode, newMode) {\n              // console.log('Ended to move to ' + newMode + ' from ' + oldMode);\n              if (newMode !== 'mode.transitioning') {\n                this.inTransitionMode = false;\n              }\n            }.bind(this)\n          );\n\n          this.sdk.on(\n            this.sdk.Sweep.Event.ENTER,\n            function (oldSweep, newSweep) {\n              // console.log('Leaving sweep ' + oldSweep);\n              // console.log('Entering sweep ' + newSweep);\n              this.inTransitionSweep = false;\n              this.displayAzimutalCrown();\n            }.bind(this)\n          );\n          this.sdk.on(\n            this.sdk.Sweep.Event.EXIT,\n            function (fromSweep, toSweep) {\n              //   console.log('Leaving sweep ' + fromSweep);\n              //   console.log('Transitioning to sweep ' + toSweep);\n              this.inTransitionSweep = true;\n            }.bind(this)\n          );\n\n          // TODO: get scene with getter instead!\n          const [sceneObject] = await this.sdk.Scene.createObjects(1);\n          const node = sceneObject.addNode();\n          node.start();\n          this.threeJSScene = node.obj3D.parent;\n\n          // await this.sdk.Scene.configure((renderer: any, three: any) => {\n          //   renderer.physicallyCorrectLights = true;\n          //   renderer.outputEncoding = three.sRGBEncoding;\n          //   renderer.shadowMap.enabled = true;\n          //   renderer.shadowMap.bias = 0.0001;\n          //   renderer.shadowMap.type = three.PCFSoftShadowMap;\n          // });\n\n\n          // TODO: wait for MP ticket resolution before decomment these!\n          // Wait until Showcase is actually playing....\n          // this.sdk.Tag.data.subscribe({\n          //   onAdded: async function (index, item, collection) {\n          //\n          //     let thisOpacity = 0.2;\n          //     this.sdk.Tag.editOpacity(index, thisOpacity);\n          //\n          //     let source = null;\n          //     try {\n          //       source = await Promise.all([\n          //         this.sdk.Sensor.createSource(this.sdk.Sensor.SourceType.SPHERE, {\n          //           origin: item.anchorPosition,\n          //           radius: Number(3),\n          //           userData: {\n          //             id: index + '-sphere-source',\n          //           },\n          //         })\n          //       ]);\n          //     } catch (e) {\n          //       console.log('could not create Sphere sensor')\n          //       console.error(e);\n          //     }\n          //     if (!source) {\n          //       return;\n          //     }\n          //\n          //     let sensor = null;\n          //     try {\n          //       sensor = await this.sdk.Sensor.createSensor(this.sdk.Sensor.SensorType.CAMERA);\n          //     } catch (e) {\n          //       console.log('could not create Camera sensor')\n          //       console.error(e);\n          //     }\n          //     if (!sensor) {\n          //       return;\n          //     }\n          //\n          //     sensor.addSource(...source);\n          //     sensor.readings.subscribe({\n          //       onUpdated(source, reading) {\n          //         console.log(thisOpacity);\n          //         let oldOpacity = thisOpacity;\n          //         if (reading.inRange) {\n          //           thisOpacity = 1;\n          //           console.log(index + ' is inRange');\n          //         } else if (reading.inView) {\n          //           console.log(index + ' is inView but not inRange');\n          //           thisOpacity = 0.5;\n          //         } else {\n          //           thisOpacity = 0.2;\n          //           console.log(index + ' is not inView or inRange');\n          //         }\n          //\n          //         this.sdk.Tag.editOpacity(index, thisOpacity);\n          //         /*\n          //                   let inc = 0.01;\n          //                   if (oldOpacity > thisOpacity) {\n          //                     inc = -0.01;\n          //                   }\n          //\n          //                   for(var i = oldOpacity; i != thisOpacity; i=i+inc) {\n          //                       setTimeout(function() {\n          //                          mpSdk.Tag.editOpacity(index, i);\n          //                       console.log('Delay', i);\n          //                       },10);\n          //                   }\n          //         */\n          //       }\n          //     });\n          //     //sensor.showDebug(true);\n          //   }.bind(this),\n          //   onCollectionUpdated: function (collection) {\n          //     console.log('Collection received. There are ', Object.keys(collection).length, ' Tags in the collection', collection);\n          //   }\n          // });\n\n          resolve(true);\n        }.bind(this)\n      );\n    });\n  }\n\n\n  setLightingOff() {\n    this.noLightForObjects = true;\n  }\n\n  pointToString(point: { x: number, y: number, z: number }): string {\n    var x = point.x.toFixed(3);\n    var y = point.y.toFixed(3);\n    var z = point.z.toFixed(3);\n\n    return `{ x: ${x}, y: ${y}, z: ${z} }`;\n  }\n\n  //\n  // ---------- Measurements related ----------\n  //\n  /**\n   * Callback after measurement is performed\n   */\n  getDistanceForLastMeasurement() {\n    if (this.lastMeasure.length > 0) {\n      const numberPoints = this.lastMeasure.length;\n      this.distancesLastMeasure = [];\n      for (let index = 1; index < numberPoints; index += 1) {\n        const distance = getDistanceBetweenTwoPoints(\n          this.lastMeasure[index - 1],\n          this.lastMeasure[index]\n        );\n        this.distancesLastMeasure.push(distance);\n      }\n      this.takeScreenShot().then((res) => {\n        this.router.navigate([`visit/${this.currentSpaceID}/add_measurement`]);\n      });\n    }\n  }\n\n  getLastDistances() {\n    return this.distancesLastMeasure;\n  }\n\n  /**\n   * Takes screenshot and saves base64 in lastScreenshotUri\n   * @returns Promise\n   */\n  takeScreenShot(): Promise<any> {\n    return this.sdk.Renderer.takeScreenShot(\n      this.resolution,\n      this.visibility\n    ).then(\n      function (screenShotUri: any) {\n        // base64 string\n        this.lastScreenshotUri = screenShotUri;\n        return Promise.resolve();\n      }.bind(this)\n    );\n  }\n\n  getScreenShotUri() {\n    return this.lastScreenshotUri;\n  }\n\n  getLastMeasurement(): Object {\n    const data = {\n      measure: this.lastMeasure,\n      sweep: this.poseCamera.sweep,\n    };\n    return data;\n  }\n\n  //\n  // ---------- Utils ----------\n  //\n  /**\n   * Styling of pointer\n   */\n  updatePointerTrick() {\n    if (\n      this.interactionMode !== ViewerInteractions.DEFAULT &&\n      this.mattertagToFollow &&\n      this.poseMatterport &&\n      this.getDistPosition(\n        this.poseMatterport.position,\n        this.oldPoseMatterportPosition\n      ) > 25\n    ) {\n      this.pointerButton.style.display = 'none';\n\n      const size = {\n        w: this.container.clientWidth,\n        h: this.container.clientHeight,\n      };\n      const coords = this.sdk.Conversion.worldToScreen(\n        this.poseMatterport.position,\n        this.poseCamera,\n        size\n      );\n      this.pointerButton.style.left = `${\n        coords.x * Math.sign(coords.x) - 25\n      }px`;\n      this.pointerButton.style.top = `${coords.y * Math.sign(coords.x) - 25}px`;\n      this.oldPoseMatterportPosition = {\n        ...this.poseMatterport.position,\n      };\n    } else {\n      this.pointerButton.style.display = 'block';\n    }\n  }\n\n  /**\n   * Realtime mattertag following the cursor\n   * @param mattertag string\n   */\n  enable_following_tag(mattertag: string) {\n    this.sdk.Tag.editPosition(mattertag, {\n      anchorPosition: {\n        x: this.poseMatterport.position.x * 1,\n        y: this.poseMatterport.position.y * 1,\n        z: this.poseMatterport.position.z * 1,\n      },\n      stemVector: {\n        x: this.poseMatterport.normal.x * 0.3,\n        y: this.poseMatterport.normal.y * 0.3,\n        z: this.poseMatterport.normal.z * 0.3,\n      },\n    });\n  }\n\n  /**\n   * Get the distance betwween two 3D positions\n   * Used in order to see how much the cursor has moved from the previous position\n   * @param pos1\n   * @param pos2\n   */\n  getDistPosition(pos1: any, pos2: any) {\n    const size = {\n      w: this.container.clientWidth,\n      h: this.container.clientHeight,\n    };\n    const coords1 = this.sdk.Conversion.worldToScreen(\n      pos1,\n      this.poseCamera,\n      size\n    );\n    const coords2 = this.sdk.Conversion.worldToScreen(\n      pos2,\n      this.poseCamera,\n      size\n    );\n    return Math.sqrt(\n      (coords1.x - coords2.x) ** 2 + (coords1.y - coords2.y) ** 2\n    );\n  }\n\n  //\n  // ---------- Mattertag related ----------\n  //\n  /**\n   * Creates the Mattertag that will follow the cursor\n   * @param mattertagData MattertagData\n   */\n  async addCursorMattertag(mattertagData: MattertagData) {\n    if (!this.poseMatterport) return;\n    this.mattertagToFollow = await this.addMattertagToViewer(mattertagData);\n    console.log('following the tag', this.mattertagToFollow);\n    this.sdk.Tag.editIcon(this.mattertagToFollow, 'icon-position');\n    this.sdk.Tag.editOpacity(this.mattertagToFollow, 1);\n  }\n\n  /**\n   * Adds Mattertag to viewer for an existing object with coordinates (in mattertagData.poi)\n   * (position, injected html, set icon)\n   * @param mattertagData\n   * returns mattertagID\n   */\n  async addMattertagToViewer(\n    mattertagData: MattertagData\n  ): Promise<string | null> {\n    let sidList;\n    if (!this.sdk) {\n      return null;\n    }\n    try {\n      sidList = await this.sdk.Tag.add(mattertagData.getData());\n    } catch (error) {\n      console.log(\n        'Tag does not belong to the visit',\n        error,\n        mattertagData.getData()\n      );\n    }\n    if (sidList) {\n      const mattertagID = sidList[0];\n      // console.log(\"added tag\", mattertagData.getType(), mattertagID);\n      this.mattertagIDs.push(mattertagID);\n      this.dictionnaryTags.set(mattertagID, mattertagData);\n      return mattertagID;\n    }\n    return null;\n  }\n\n  /**\n   * Actions when position of mattertag is validated by left click\n   */\n  onValidatedMattertag() {\n    this.setInteractionMode(ViewerInteractions.DEFAULT);\n    const lastTag = this.getLastTag();\n    if (lastTag) {\n      const callbackMode = this.dictionnaryTags\n        .get(lastTag)\n        .getCallbackActionMode();\n      this.callbackAfterMattertagValidation(callbackMode);\n    }\n  }\n\n  /**\n   * Registers new icon (path to image) and set its for Mattertag\n   * @param mattertagID string\n   * @param iconPath string\n   */\n  async addNewIconAndSetForTag(mattertagID: string, iconPath: string) {\n    await this.sdk.Asset.registerTexture(`icon_${mattertagID}`, iconPath);\n    this.sdk.Tag.editIcon(mattertagID, `icon_${mattertagID}`);\n  }\n\n  /**\n   * Changes icon of Mattertag (the iconName should be registered = one of default ones)\n   * @param mattertagID string\n   * @param iconName string\n   */\n  async setRegistredIconForTag(mattertagID: string, iconName: string) {\n    try {\n      this.sdk.Tag.editIcon(mattertagID, iconName);\n    } catch (e) {\n      console.log(\n        'could not edit Icon with name ',\n        iconName,\n        '. Is it registered?'\n      );\n    }\n  }\n\n  /**\n   * Sets default icon for a tag (registered in initSdk) OR uses tagIcon from POI (available from MattertagData)\n   * @param mattertagID string\n   * @param mattertagData MattertagData\n   * @returns\n   */\n  async setTagIconAndOpacity(\n    mattertagID: string,\n    mattertagData: MattertagData\n  ) {\n    if (this.SPModule === SpModule.HOTEL) {\n      const room = mattertagData.getObject();\n      let iconName = `icon_room`;\n      if (room.available) {\n        iconName = `${iconName}_available`;\n      } else {\n        iconName = `${iconName}_unavailable`;\n      }\n      this.sdk.Tag.editIcon(mattertagID, iconName);\n      return;\n    }\n    const stringTagType = poiTypeToString(mattertagData.getType());\n    let opacity = this.config.my_config.DEFAULT_OPACITY_TAG;\n    let iconName = `icon-${stringTagType}`;\n    const poi = mattertagData.getPoi();\n    if (poi && poi.tagIcon) {\n      const tagIcon = JSON.parse(poi.tagIcon);\n      if (mattertagData.getType() === PoiType.DATA) {\n        const feature = mattertagData.getObject();\n        if (feature.type === FeatureType.INDICATOR_TEMP) {\n          tagIcon.src = this.tagService.getIconTagImageForFeature(feature, poi);\n        }\n      }\n      if (tagIcon.src) {\n        const pathSigned = await this.tagService.getSignedTagIconSource(\n          tagIcon.src\n        );\n        if (pathSigned) {\n          try {\n            iconName = `icon-${stringTagType}-${mattertagID}-${mattertagData.customIconIndex}`;\n            await this.sdk.Asset.registerTexture(iconName, pathSigned);\n            mattertagData.customIconIndex += 1;\n          } catch {\n            console.log('error registerIcon');\n          }\n        }\n      }\n      if (tagIcon.opacity) {\n        opacity = tagIcon.opacity;\n      }\n    }\n    try {\n      this.sdk.Tag.editIcon(mattertagID, iconName);\n      this.sdk.Tag.editOpacity(mattertagID, opacity);\n    } catch (error) {\n      console.log('Error editIcon or opacity', error);\n    }\n  }\n\n  /**\n   * Moves viewer to last tag created\n   */\n  goToLastTag() {\n    setTimeout(() => {\n      const lastTag = this.getLastTag();\n      this.goToTag(lastTag);\n    }, 2000);\n  }\n\n  /**\n   * Moves viewer to Mattertag with ID provided\n   * @param mattertagID string\n   * @returns\n   */\n  async goToTag(mattertagID: string) {\n    if (mattertagID === '') return;\n    try {\n      this.onGoToTag.next(mattertagID);\n      await this.sdk.Sweep.current.waitUntil(\n        (currentSweep) => currentSweep !== ''\n      );\n      await this.sdk.Mattertag.navigateToTag(\n        mattertagID,\n        this.sdk.Mattertag.Transition.FLY\n      );\n    } catch (error) {\n      console.log('cannot navigate to tag', error);\n    }\n  }\n\n  /**\n   * Updates content of Mattertag with mattertagID (billboard, injected html, tag icon)\n   * @param mattertagID string\n   * @param object Ticket, Equipment, Feature, etc\n   * @param tagType PoiType\n   */\n  async updateMatterTagContentForTagID(\n    mattertagID: string,\n    object: DbObjectType = null,\n    tagType: PoiType = null\n  ) {\n    this.sdk.Tag.editBillboard(\n      mattertagID,\n      this.dictionnaryTags.get(mattertagID).getData()\n    );\n    if (object && tagType) {\n      await this.injectHtmlInTag(tagType, object, mattertagID);\n    }\n    await this.setTagIconAndOpacity(\n      mattertagID,\n      this.dictionnaryTags.get(mattertagID)\n    );\n  }\n\n  /**\n   * Updates injected html for Mattertag\n   * @param mattertagID string\n   * @param object Ticket, Equipment, Feature, etc\n   * @param tagType PoiType\n   */\n  async updateInjectedHtmlForTagID(\n    mattertagID: string,\n    object: any,\n    tagType: PoiType\n  ) {\n    await this.injectHtmlInTag(tagType, object, mattertagID);\n  }\n\n  /**\n   * Deletes Mattertag from Viewer by its ID\n   * @param mattertagID string\n   */\n  deleteMattertagFromId(mattertagID: string) {\n    this.sdk.Tag.remove(mattertagID);\n    this.dictionnaryTags.delete(mattertagID);\n  }\n\n  /**\n   * Deletes latest created mattertag\n   */\n  deleteLastMattertag() {\n    const mattertagID = this.mattertagIDs.pop();\n    if (mattertagID) {\n      this.deleteMattertagFromId(mattertagID);\n    }\n  }\n\n  /**\n   * Legacy: used to be called action_add_mattertag_from_POI\n   * Adds and configures Mattertag for an object (Ticket, Equipment, Feature, etc) that corresponds to POI (coordinates, tagIcon)\n   * @param tagType PoiType\n   * @param object Ticket, Equipment, Feature...\n   * @param poi POI\n   * @returns\n   */\n  async createMattertagFromPOI(\n    tagType: PoiType,\n    object: DbObjectType,\n    poi: POI\n  ): Promise<any> {\n    // check if tag exists already\n    const {tag, sweep} = this.getTagFromElementId(object.id);\n    if (tag) {\n      // console.log(\"tag exists\", object)\n      return;\n    }\n    const mattertagData: MattertagData = new MattertagData(tagType);\n    mattertagData.setObject(object, tagType);\n    if (poi.coordinate) {\n      mattertagData.setPosition(JSON.parse(poi.coordinate));\n    }\n    if (poi.metadata) {\n      const tagMetadata = JSON.parse(poi.metadata);\n      if (tagMetadata.normal) {\n        mattertagData.setNormal(tagMetadata.normal);\n      } else {\n        mattertagData.setNormal({x: 0, y: -0.15, z: 0});\n      }\n    }\n    if (poi.matterportSweepID) {\n      mattertagData.setSweepID(poi.matterportSweepID);\n    }\n    mattertagData.setPoi(poi);\n\n    const createdTagID = await this.addMattertagToViewer(mattertagData);\n    if (createdTagID && this.sdk) {\n      await this.setTagIconAndOpacity(createdTagID, mattertagData);\n      await this.injectHtmlInTag(tagType, object, createdTagID);\n    }\n  }\n\n  /**\n   * Inject custom HTML as Mattertag content\n   * @param tagType PoiType\n   * @param object Ticket, Equipment, Feature etc\n   * @param tagID string\n   */\n  async injectHtmlInTag(tagType: PoiType, object: DbObjectType, tagID: string) {\n    let html = await this.tagService.getHtmlToInject(tagType, object);\n    if (html !== '' && this.sdk) {\n      const scriptTag = this.tagService.getScriptForTag(object, tagType);\n      html += `${scriptTag}`;\n      // create and register the sandbox\n      const [sandboxId, messenger] = await this.sdk.Tag.registerSandbox(html);\n\n      // detach previous sandbox from a tag\n      let attachmentID = this.tagsAttachments[tagID];\n      if (attachmentID) {\n        this.sdk.Tag.detach(tagID, attachmentID);\n      }\n      this.tagsAttachments[tagID] = sandboxId;\n\n      // attach the sandbox to a tag\n      this.sdk.Tag.attach(tagID, sandboxId);\n      // receive data from the sandbox\n      // tagMessengerOn allows to go here only once\n      if (!this.tagMessengerOn) {\n        this.tagMessengerOn = true;\n        const imageClick = (featureID: string) => {\n          // console.log(\"image click handler\", featureID);\n          this.tagService.onActionImageClick(featureID);\n        };\n\n        const audioClick = (audioCommentID: string) => {\n          // console.log(\"audio click handler\", audioCommentID);\n          this.tagService.onActionAudioClick(audioCommentID);\n        };\n        const videoClick = (url: string) => {\n          this.tagService.onActionVideoClick(url);\n        };\n        messenger.on(\n          TagAction.DETAIL_CLICK,\n          this.tagService.onActionDetailClick.bind(this.tagService)\n        );\n        messenger.on(\n          TagAction.TICKET_CLICK,\n          this.tagService.onActionDetailClick.bind(this.tagService)\n        );\n        messenger.on(TagAction.AUDIO_CLICK, audioClick);\n        messenger.on(TagAction.IMAGE_CLICK, imageClick);\n        messenger.on(TagAction.VIDEO_CLICK, videoClick);\n        messenger.on(\n          TagAction.DOC_CLICK,\n          this.tagService.onActionDocClick.bind(this.tagService)\n        );\n        messenger.on(\n          TagAction.YOUTUBE_CLICK,\n          this.tagService.onActionYoutubeClick.bind(this.tagService)\n        );\n      }\n    } else {\n      // if html is empty (case of EMBED content), we edit billboard for Feature and attach new content\n      const {comment, tagDescription} =\n        this.tagService.getBillboardMediaToEmbed(object);\n      if (comment) {\n        // register the media\n        const [attachmentID] = await this.sdk.Tag.registerAttachment(\n          comment.externalLink\n        );\n        // attach\n        this.sdk.Tag.attach(tagID, attachmentID);\n        this.sdk.Tag.editBillboard(tagID, {\n          label: object.title,\n          description: tagDescription,\n        });\n      }\n    }\n  }\n\n  async action_delete_all_mattertags() {\n    await this.sdk.Tag.remove(...this.mattertagIDs);\n    this.mattertagIDs = [];\n    this.dictionnaryTags.clear();\n  }\n\n  /**\n   * Deletes Mattertag from visit associated with object ID (ticketID, etc)\n   * @param elementID string\n   */\n  async deleteMattertagForObject(elementID: string) {\n    const matterTagID = this.getTagFromElementId(elementID).tag;\n    if (matterTagID) {\n      try {\n        await this.sdk.Tag.remove(matterTagID);\n        this.dictionnaryTags.delete(matterTagID);\n        const index = this.mattertagIDs.indexOf(matterTagID);\n        this.mattertagIDs.splice(index, 1);\n      } catch (error) {\n        console.log('Cannot delete tag', matterTagID, error);\n      }\n    }\n  }\n\n  /**\n   * uuid from threejs\n   * @param uuid\n   */\n  async deleteObject3D(uuid: string): Promise<void> {\n    this.dictionnaryObjects3D.get(uuid).obj3D.visible = false;\n  }\n\n  getObject3DModelNodeFromDictionnary(uuid: string): any {\n      let obj = this.dictionnaryObjects3D.get(uuid);\n      if(!obj){\n        console.log(\"weird thing again\");\n        return null;\n      }\n      //might break things or fix everything ..?\n      if(obj.obj3D.uuid != uuid){\n        console.log(\"we have THE problem\");\n        obj.obj3D.uuid = uuid; //not enugh to fix the pb\n      }\n\n      return obj;\n  }\n\n  /**\n   * Creates  MattertagData and start repositioning (creates temporary mattertag that follows the cursor)\n   * @param poiType PoiType\n   * @param element Ticket, Equipment, Feature, etc\n   */\n  async addMattertagWhenRepositioning(poiType: PoiType, element: DbObjectType) {\n    const mattertagData = new MattertagData(poiType);\n    // set the coordinates of the existing tag\n    const [poi] = element.pois.items;\n    if (poi && poi.coordinate) {\n      mattertagData.setPosition(JSON.parse(poi.coordinate));\n      mattertagData.setPoi(poi); // to keep custom tagIcon and opacity\n    }\n    mattertagData.setSweepID(this.poseCamera.sweep);\n    mattertagData.setRotation(this.poseCamera.rotation);\n    mattertagData.setObject(element, poiType);\n    this.mattertagToFollow = await this.addMattertagToViewer(mattertagData);\n    this.setInteractionMode(ViewerInteractions.POSITIONING);\n    await this.addCursorMattertag(mattertagData);\n  }\n\n  /**\n   * Creates MattertagData and mattertag that follows the cursor when choosing position for a new object\n   * @param poiType\n   */\n  async addMattertagWhenAdding(poiType: PoiType) {\n    const mattertagData = new MattertagData(poiType);\n    mattertagData.setSweepID(this.poseCamera.sweep);\n    mattertagData.setRotation(this.poseCamera.rotation);\n    this.setInteractionMode(ViewerInteractions.ADDING);\n    await this.addCursorMattertag(mattertagData);\n  }\n\n  /**\n   * Cancels following of cursor (meaning deleting last Mattertag)\n   */\n  cancelFollowingCursor() {\n    this.deleteLastMattertag();\n    this.setInteractionMode(ViewerInteractions.DEFAULT);\n  }\n\n  setLastObject3D(lastObject3D: ThreeJSObject3D) {\n    this.lastObject3D = lastObject3D;\n  }\n\n  getLastObject3D(): ThreeJSObject3D {\n    return this.lastObject3D;\n  }\n\n  /**\n   * Performs callback after mattertag position was validated (creation of object or repositioning)\n   * @param mode MattertagActionMode\n   */\n  callbackAfterMattertagValidation(mode: MattertagActionMode) {\n    this.visibilityService.detailShowing.next(true);\n    const lastTag = this.getLastTag();\n    switch (mode) {\n      case MattertagActionMode.ADD_EQUIPMENT:\n        this.goToLastTag();\n        this.router.navigate([`visit/${this.currentSpaceID}/add_equip`]);\n        break;\n      case MattertagActionMode.ADD_TICKET:\n        this.goToLastTag();\n        this.router.navigate([`visit/${this.currentSpaceID}/add_ticket`]);\n        break;\n      case MattertagActionMode.ADD_OBJECT3D:\n        this.goToLastTag();\n        this.router.navigate([`visit/${this.currentSpaceID}/add_object3d`]);\n        break;\n      case MattertagActionMode.ADD_DATA:\n        this.goToLastTag();\n        let url: string;\n        if (this.router.url.includes('?')) {\n          url = this.router.url.substring(0, this.router.url.indexOf('?'));\n        } else {\n          url = this.router.url;\n        }\n        this.router.navigate([`${url}/add_feature`]);\n        break;\n      case MattertagActionMode.ADD_DESK:\n        this.goToLastTag();\n        this.router.navigate([`visit/${this.currentSpaceID}/add_feature`], {\n          queryParams: {isDesk: true},\n        });\n        break;\n      case MattertagActionMode.ADD_ROOM:\n        this.goToLastTag();\n        this.router.navigate([`visit/${this.currentSpaceID}/add_room`]);\n        break;\n      case MattertagActionMode.POSITION_OBJECT3D:\n        if (lastTag) {\n          // DO Nothing (routing), just\n          this.router.navigate(\n            [`visit/${this.currentSpaceID}/object3d/${this.lastObject3D.id}`],\n            {queryParams: {positioning: true}}\n          );\n        }\n        break;\n      case MattertagActionMode.POSITION_TICKET:\n        if (lastTag) {\n          this.router.navigate(\n            [\n              `visit/${this.currentSpaceID}/detail/${\n                this.dictionnaryTags.get(lastTag).elementID\n              }`,\n            ],\n            {queryParams: {positioning: true}}\n          );\n        }\n        break;\n      case MattertagActionMode.POSITION_EQUIPMENT:\n        if (lastTag) {\n          this.router.navigate(\n            [\n              `visit/${this.currentSpaceID}/equip/${\n                this.dictionnaryTags.get(lastTag).elementID\n              }`,\n            ],\n            {queryParams: {positioning: true}}\n          );\n        }\n        break;\n      case MattertagActionMode.POSITION_DATA:\n        if (lastTag) {\n          this.router.navigate([], {\n            relativeTo: this.activeRoute,\n            queryParams: {positioning: true}, queryParamsHandling : \"merge\"\n          });\n        }\n        break;\n      case MattertagActionMode.POSITION_ROOM:\n        if (lastTag) {\n          this.router.navigate([], {\n            relativeTo: this.activeRoute,\n            queryParams: {positioning: true}, queryParamsHandling : \"merge\"\n          });\n        }\n        break;\n    }\n  }\n\n  /**\n   * Fully updates existing Mattertag content with data of object (Ticket, Equipment, Desk)\n   * @param mattertagID string\n   * @param object Ticket, Equipment, Feature, Desk\n   * @param poiType PoiType\n   * @param poi POI\n   */\n  async setObjectAndPoiInTag(\n    mattertagID: string,\n    object: DbObjectType,\n    poiType: PoiType,\n    poi: POI | null = null\n  ): Promise<void> {\n    this.dictionnaryTags.get(mattertagID).setObject(object, poiType);\n    if (poi) {\n      this.dictionnaryTags.get(mattertagID).setPoi(poi);\n    }\n    this.dictionnaryTags.get(mattertagID).elementID = object.id;\n    try {\n      await this.updateMatterTagContentForTagID(mattertagID, object, poiType);\n      // TODO: fix this\n      await this.updateMatterTagPosInSdkViewer(\n        mattertagID,\n        object,\n        poiType,\n        poi\n      );\n    } catch (e) {\n      console.log(`error in setObjectAndPoiInTag: ${e}`);\n    }\n  }\n\n  async updateMatterTagPosInSdkViewer(\n    mattertagID: string,\n    object: DbObjectType,\n    poiType: PoiType,\n    poi: POI | null = null\n  ): Promise<void> {\n    const IndexToUpdate = object.pois.items.findIndex((u) => u.id === poi.id);\n    object.pois.items[IndexToUpdate] = poi;\n    const {x, y, z} = JSON.parse(poi.coordinate);\n    const newPosition = {\n      anchorPosition: {\n        x: x,\n        y: y,\n        z: z,\n      },\n      stemVector: {\n        // make the Tag stick straight up and make it 0.30 meters (~1 foot) tall\n        x: 0,\n        y: 0.3,\n        z: 0,\n      },\n    };\n    this.sdk.Mattertag.editPosition(mattertagID, newPosition);\n    this.sdk.Tag.editPosition(mattertagID, newPosition);\n    this.dictionnaryTags\n      .get(mattertagID)\n      .setPosition(newPosition.anchorPosition);\n\n    for (let tagId of this.mattertagIDs) {\n      const currentTag = this.dictionnaryTags.get(tagId);\n      if (!currentTag) {\n        continue;\n      }\n      if (currentTag.elementID === object.id && tagId !== mattertagID) {\n        this.dictionnaryTags.delete(tagId);\n        this.sdk.Tag.remove(tagId);\n        this.sdk.Mattertag.remove(tagId);\n        break;\n      }\n    }\n  }\n\n  /**\n   * Gets matterTagID and its sweep for an object ID (ticket ID, etc)\n   * @param elementID string\n   * @returns {tag: string | null, sweep: string | null}\n   */\n  public getTagFromElementId(elementID: string): {\n    tag: string | null;\n    sweep: string | null;\n  } {\n    let tagID = null;\n    let sweepID = null;\n    for (let [mattertagID, mattertagData] of this.dictionnaryTags) {\n      if (mattertagData.elementID === elementID) {\n        tagID = mattertagID;\n        const sweep = mattertagData.getSweepID();\n        if (sweep && this.sweeps && this.sweeps.includes(sweep)) {\n          sweepID = sweep;\n        }\n      }\n    }\n    return {tag: tagID, sweep: sweepID};\n  }\n\n  /**\n   * Gets latest tag created in the visit (when following the cursor to position)\n   * @returns string mattertagID\n   */\n  getLastTag(): string | null {\n    if (this.mattertagToFollow) {\n      return this.mattertagToFollow;\n    } else {\n      return this.mattertagIDs.length === 0\n        ? null\n        : this.mattertagIDs[this.mattertagIDs.length - 1];\n    }\n  }\n\n  /**\n   * Gets MattertagData for Mattertag (replaces getTagDataFromId)\n   * @param mattertagID\n   * @returns MattertagData\n   */\n  getMatterTagDataForMattertag(mattertagID: string): MattertagData | null {\n    try {\n      return this.dictionnaryTags.get(mattertagID);\n    } catch {\n      console.log('cannot retrieve mattertagData for tag', mattertagID);\n    }\n    return null;\n  }\n\n  //\n  // ---------- Viewer related (switch views, go to) ----------\n  //\n  async action_toolbox_floorplan() {\n    if (this.inTransitionMode || this.inTransitionSweep) {\n      console.log('viewer is in transition, cannot go floorplan');\n      return;\n    }\n    try {\n      await this.sdk.Mode.moveTo('mode.floorplan');\n    } catch (e) {\n      console.log('cannot go to floorplan', e);\n    }\n  }\n\n  action_toolbox_inside_view() {\n    this.sdk.Mode.moveTo('mode.inside');\n  }\n\n  actionShowAllFloors() {\n    try {\n      this.sdk.Floor.showAll();\n    } catch (e) {\n      console.log('cannot show all floors', e);\n    }\n  }\n\n  async action_toolbox_dollhouse() {\n    setTimeout(async () => {\n      // console.log(\"mode: \", this.inTransitionMode, \" sweep: \", this.inTransitionSweep);\n      if (this.inTransitionMode || this.inTransitionSweep) {\n        console.log('viewer is in transition, cannot go dollhouse');\n        return;\n      }\n      try {\n        await this.sdk.Mode.moveTo('mode.dollhouse');\n      } catch (e) {\n        console.log('cannot go to dollhouse', e);\n      }\n    }, 1200);\n  }\n\n  action_toolbox_mesure() {\n    const newState = !this.isMeasureModeOn;\n    this.sdk.Measurements.toggleMode(newState).then(() => {\n      console.log(\n        `Measurement mode is now ${newState ? 'enabled' : 'disabled'}`\n      );\n    });\n  }\n\n  action_toolbox_cancel_mesure() {\n    if (this.isMeasureModeOn) {\n      this.action_toolbox_mesure();\n    }\n  }\n\n  async action_go_to_floor(\n    floorName: string,\n    matterportFloorSequence: number = null\n  ) {\n    if (!this.floors) {\n      console.log('Floor are not loaded yet');\n      return;\n    }\n    // console.log(this.floors);\n    // look up for sequence number (the safest method)\n    let floorMatterport = this.floors.find(\n      (floor) => floor.sequence === matterportFloorSequence\n    );\n    if (!floorMatterport) {\n      floorMatterport = this.floors.find(\n        (floor) => floorName.includes(floor.name) && floor.name != ''\n      );\n    }\n    if (!floorMatterport) {\n      floorMatterport = this.floors.find((floor) =>\n        floorName.includes(floor.id)\n      );\n    }\n    // console.log(floorMatterport)\n    if (floorMatterport) {\n      let retry = true;\n      while (retry) {\n        try {\n          const floorIndex = await this.sdk.Floor.moveTo(\n            floorMatterport.sequence\n          );\n          // console.log(\"moved to floorIndex\", floorIndex);\n          retry = false;\n        } catch (error) {\n          console.log('Cannot move to Floor', error);\n          await wait(100);\n        }\n      }\n    } else {\n      console.warn('No matterport floor found to move to');\n    }\n  }\n\n  async action_go_to_sweep(\n    sweep: string,\n    rotation: { x: number; y: number } = null\n  ) {\n    if (this.forbiddenSweeps.includes(sweep)) {\n      console.log('user is not allowed to go to this sweep');\n      return;\n    }\n    // console.log(\"going to sweep\", sweep, \"with rotation:\", rotation);\n    setTimeout(async () => {\n      if (this.inTransitionMode || this.inTransitionSweep) {\n        console.log('Cannot go to sweep, in transition');\n        return;\n      }\n      try {\n        this.inTransitionSweep = true;\n        await this.sdk.Sweep.moveTo(sweep, {\n          transition: 'transition.instant',\n          transitionTime: 1500,\n        });\n      } catch (error) {\n        this.inTransitionSweep = false;\n        console.log('Cannot move to sweep', error);\n      }\n      if (rotation) {\n        await this.sdk.Camera.setRotation(rotation, {speed: 100}); // speed is degrees per second\n      }\n    }, 1000);\n  }\n\n  getCurrentSweep(): string | null {\n    if (this.poseCamera) {\n      return this.poseCamera.sweep;\n    }\n    return null;\n  }\n\n  getCurrentCameraPosition(): {\n    rotation: any;\n    position: any;\n    sweep: any;\n  } | null {\n    if (this.poseCamera) {\n      return this.poseCamera;\n    }\n    return null;\n  }\n\n  setInteractionMode(interactionMode: number) {\n    this.interactionMode = interactionMode;\n  }\n\n  getInteractionMode() {\n    return this.interactionMode;\n  }\n\n  /**\n   * Clear all variables, deletes all tags (when viewer is remover or model changed=reload needed)\n   */\n  async clearAll() {\n    console.log('removing viewer');\n    this.action_delete_all_mattertags();\n    this.floors = null;\n    this.sweeps = null;\n    this.sdk = null;\n    clearInterval(this.timerPointer);\n    this.forbiddenSweeps = [];\n    this.tagMessengerOn = false;\n    // cancel subscriptions\n    this.pointerButton.removeEventListener(\n      'click',\n      this.pointerLeftClickHandler\n    );\n    this.pointerButton.removeEventListener(\n      'contextmenu',\n      this.pointerRightClickHandler\n    );\n\n    // TODO: only For Admins\n    if (this.getCursorPositionButton) {\n      this.getCursorPositionButton.removeEventListener(\n        'auxclick',\n        this.pointerMiddleClickHandler\n      );\n    }\n    // TODO: only for dev!\n    if (!!this.getCursorPositionButton &&\n      (document.location.href.indexOf('dev') !== -1 || document.location.href.indexOf('localhost') !== -1)) {\n      clearInterval(this.intervalCursorPointerPosition);\n    }\n  }\n\n  async removeForbiddenSweeps(forbiddenSweeps: string[]) {\n    this.forbiddenSweeps = [...forbiddenSweeps];\n    let removed = 0;\n    await Promise.all(\n      forbiddenSweeps.map(async (sweep) => {\n        try {\n          await this.sdk.Sweep.disable(sweep);\n          removed += 1;\n        } catch (error) {\n          console.log(error);\n        }\n      })\n    );\n    console.log('removed sweeps:', removed);\n  }\n\n  //\n  // ---------- 3D objects (SDK Bundle only) ----------\n  //\n  async init3DObjectViewer(): Promise<void> {\n    return new Promise(async (resolve) => {\n      var [sceneObject] = await this.sdk.Scene.createObjects(1);\n\n      var node = sceneObject.addNode();\n\n      // TODO change this 🤮\n      node.addComponent('mp.lights');\n      //node.addComponent('mp.lights');\n      /*node.addComponent('mp.lights');\n      node.addComponent('mp.lights');*/\n      node.start();\n\n      const nodeControl = sceneObject.addNode();\n      this.objectControl = nodeControl.addComponent('mp.transformControls');\n      nodeControl.start();\n\n      //this.add3DObject({},null);\n\n      resolve();\n    });\n  }\n\n  async add3DObject(\n    obj: IObject3D,\n    mode?: 'translate' | 'rotate'\n  ): Promise<any> {\n    return new Promise(async (resolve) => {\n      const [sceneObject] = await this.sdk.Scene.createObjects(1);\n\n      // TODO: improvment, regroup all of these in Dynamical Objects Lib\n      // SecurityCamera\n      // NestThermostat\n      // Video\n      let isAnimatedSecurityCamera = false;\n      let isNestThermostat = false;\n      let isSmarterplanPromotionalVideo = false;\n      let isAzimuthalCrown = false;\n\n      /**\n       * TODO: refacto with an enum or switch/case\n       */\n      if (obj.object === \"security_camera\") {\n        isAnimatedSecurityCamera = true;\n      }\n      if (obj.object === \"nest_thermostat\") {\n        isNestThermostat = true;\n      }\n      if (obj.object === 'video') {\n        isSmarterplanPromotionalVideo = true;\n      }\n      if(obj.object === 'azimuth') {\n        isAzimuthalCrown = true;\n      }\n\n      const modelNode = sceneObject.addNode();\n      let component = null;\n      const initial = {\n        url: `/assets/3Dobjects/objects/${obj.object}${\n          obj.format.indexOf('.') === -1 ? '.' + obj.format : obj.format\n        }`,\n        // TODO/ store localPosition && localRotation in BDD too (in order to have pertfect initial placement)\n        localRotation: {\"x\": 0, \"y\": 0, \"z\": 0},\n        // TODO/ store localPosition && localRotation in BDD too (in order to have pertfect initial placement)\n        localPosition: {\"x\": 0, \"y\": 0, \"z\": 0},\n        visible: true,\n        colliderEnabled: true\n      };\n\n\n      switch (obj.format) {\n        case '.obj':\n        case 'obj':\n          component = modelNode.addComponent('mp.objLoader', initial);\n          break;\n        case '.fbx':\n        case 'fbx':\n          component = modelNode.addComponent('mp.fbxLoader', initial);\n          break;\n        case '.gltf':\n        case 'gltf':\n          component = modelNode.addComponent('mp.gltfLoader', initial);\n          break;\n        case '.glb':\n        case 'glb':\n          component = modelNode.addComponent('mp.gltfLoader', initial);\n          break;\n        case '.dae':\n        case 'dae':\n          component = modelNode.addComponent('mp.daeLoader', initial);\n          break;\n\n        default:\n          console.log('Format not supported');\n          break;\n      }\n\n      //cache system (i'll try to make it work later ...)\n      /*let objContentsJSON = JSON.stringify(modelNode);\n      console.log(modelNode);\n      console.log(objContentsJSON);\n      console.log(JSON.stringify(modelNode));\n      localStorage.setItem(`testObj_${obj.object}`, objContentsJSON);\n      console.log(\"stored ?!\");\n      console.log(typeof modelNode);*/\n      //let dataS = serialijse.serialize(modelNode);\n      //console.log(dataS);\n\n      // Use 3 ?! Ambient lightings\n      if (this.noLightForObjects) {\n        const lightsNode = sceneObject.addNode();\n        lightsNode.addComponent('mp.ambientLight', {\n          intensity: 1,\n          color: {r: 1.0, g: 1.0, b: 1.0},\n        });\n        this.noLightForObjects = false;\n      }\n\n      modelNode.obj3D.position.set(\n        obj.position.x,\n        obj.position.y,\n        obj.position.z\n      );\n\n      modelNode.obj3D.rotation.set(\n        obj.rotation.x,\n        obj.rotation.y,\n        obj.rotation.z\n      );\n      modelNode.obj3D.scale.set(obj.scale.x, obj.scale.y, obj.scale.z);\n      if(isAzimuthalCrown)\n      {\n        /*if(modelNode.obj3D.scale.x >= 1.0){\n          modelNode.obj3D.scale.set(obj.scale.x/5000, obj.scale.y/5000, obj.scale.z/5000);\n        }*/\n        this.azimuthalCrown=modelNode;\n        this.displayAzimutalCrown();\n      }\n\n      // By defaut, during creation, object has translation gizmo\n      // => User has to click on lateral panel, and save its position after playing with it!\n      if (mode && !isNestThermostat) {\n        this.attachGizmoControlTo3DObject(\n          modelNode,\n          sceneObject,\n          mode,\n          true,\n          true\n        ).catch((e) => console.log(e));\n      }\n\n      if (this.lastObject3D && typeof this.lastObject3D.id === 'string') {\n        // prompt ThreeJS UUID to match our last generated object if new (not from Db)\n        modelNode.obj3D.uuid = this.lastObject3D.id;\n      } else if (obj.id) {\n        modelNode.obj3D.uuid = obj.id;\n      }\n      this.lastObject3D = modelNode.obj3D;\n\n      // Store this in memory Map dictionnary\n      //console.log(\"Adding object: \");\n      //console.log(modelNode);\n      this.dictionnaryObjects3D.set(modelNode.obj3D.uuid, modelNode);\n      this.dictionnarySceneObjects3D.set(modelNode.obj3D.uuid, sceneObject);\n\n      /*this.sdk.Camera.pose.subscribe(\n        function (pose: any) {\n          //console.log(\"in callback\")\n          //console.log(this.lastCameraPosition)\n          //console.log(pose.position)\n          if((pose.position.x == this.lastCameraPosition.x && pose.position.y == this.lastCameraPosition.y && pose.position.z == this.lastCameraPosition.z) || this.inTransitionMode || this.inTransitionSweep){\n            //console.log(\"returning\")\n            return;\n          }\n          console.log(\"camera pos:\",pose.position);\n          this.lastCameraPosition = {...pose.position};\n          if(this.lastObject3D){\n            this.lastObject3D.position.set(pose.position.x+.5,pose.position.y+.5,pose.position.z+.5);\n          }\n        }.bind(this));*/\n\n      if (isAnimatedSecurityCamera) {\n\n        const animator = modelNode.addComponent('mp.securityCamera',\n          {\n            \"nearPlane\": 0.1,\n            \"farPlane\": 10,\n            \"horizontalFOV\": 52,\n            \"aspect\": 1.7777777777777777,\n            \"localPosition\": {\n              \"x\": 0.3,\n              \"y\": 0.18,\n              \"z\": 0\n            },\n            \"localRotation\": {\n              \"x\": -15,\n              \"y\": -90,\n              \"z\": 0\n            },\n            \"color\": 65280,\n            \"panPeriod\": 5,\n            \"panAngle\": -45\n          });\n        const modelOutput = sceneObject.addPath({\n          id: 'animated-model',\n          type: this.sdk.Scene.PathType.OUTPUT,\n          node: modelNode,\n          component: animator,\n          property: 'objectRoot'\n        });\n\n        this.securityCameraAnimator = animator;\n        if (!obj.viewFrustum) {\n          setTimeout(() => {\n            animator.toggleViewFrustum();\n          }, 1000);\n        }\n      }\n\n      if (isNestThermostat) {\n        // TODO: use bindPath instead using MP sdk classes (see Security Camera example)\n        // for TV uses CanvasImage below!\n        // const ci = new CanvasImage();\n        // ci.onInit();\n        const cv = new CanvasRenderer();\n        cv.onInit();\n        const plane = new PlaneRenderer();\n        plane.outputs = {\n          objectRoot: new Object3D(),\n          collider: new Object3D()\n        };\n        const inputTexture = cv.outputs.texture;\n        plane.setRootScene(this.threeJSScene);\n        plane.onInit(modelNode, inputTexture);\n        const sc = new NestThermostat();\n        sc.setComponent(component, plane, cv);\n        sc.setRootScene(this.threeJSScene);\n        sc.onInit(modelNode, plane, inputTexture);\n        cv.setCanvasNestThermostatPainter(sc);\n        setTimeout(() => {\n          sc.inputs.loadingState = \"Loaded\";\n          sc.onInputsUpdated();\n        }, 5000);\n      }\n\n      if (isSmarterplanPromotionalVideo) {\n        // TODO: use bindPath instead using MP sdk classes (see Security Camera example)\n        const sc = new TvPlayer();\n        sc.setComponent(component);\n        sc.onInit(modelNode);\n        setTimeout(() => {\n          sc.inputs.loadingState = \"Loaded\";\n          sc.onInputsUpdated();\n        }, 5000);\n      }\n\n      sceneObject.start();\n      resolve(this.lastObject3D);\n    });\n  }\n\n  toggleObjectVisibility(objectId: any){\n    let obj = this.dictionnaryObjects3D.get(objectId);\n    obj.obj3D.visible = !obj.obj3D.visible;\n  }\n\n  isObjectVisible(objectId: any): boolean {\n    let obj = this.dictionnaryObjects3D.get(objectId);\n    return obj.obj3D.visible;\n  }\n\n  async pointCameraTo3DObject(objectId: any){\n    let obj = this.dictionnaryObjects3D.get(objectId);\n\n    //We create a temporary Tag\n    const poiObject = {\n      coordinate: JSON.stringify(obj.obj3D.position),\n      type: PoiType.OBJECT3D,\n      elementID: objectId, //todo: be careful with this\n    } as POI;\n    const objectDb = {id: objectId} as DbObjectType;\n\n    try{\n      await this.createMattertagFromPOI(PoiType.OBJECT3D, objectDb, poiObject);\n    }catch(err: any){}\n\n    //Not really necessary anymore since the tag will disappear quick\n    //this.sdk.Tag.editOpacity(mattertagID, 0.0);//opacity);\n    //this.sdk.Tag.allowAction(mattertagID, {}); //disables every action\n\n    let result = this.getTagFromElementId(objectId);\n\n    await this.goToTag(result.tag);\n    this.deleteLastMattertag();\n  }\n\n  getSceneNodeFromObject3DId(uuid: string): any {\n    return this.dictionnarySceneObjects3D.get(uuid);\n  }\n\n  async displayAzimutalCrown(){\n    if(this.azimuthalCrown){\n      this.azimuthalCrown.obj3D.position.set(this.poseCamera.position.x,this.poseCamera.position.y,this.poseCamera.position.z)\n    }\n  }\n\n  async attachGizmoControlTo3DObject(\n    modelNode: any,\n    sceneObject: any,\n    mode: 'translate' | 'rotate' | 'scale',\n    visible: boolean,\n    isNewObject: boolean\n  ): Promise<any> {\n    // Create a scene node with a transform control component.\n    let node = null;\n    node = sceneObject.addNode();\n    if (!node) {\n      const [sceneObject] = await this.sdk.Scene.createObjects(1);\n      node = sceneObject.addNode();\n    }\n\n    const myControl = node.addComponent('mp.transformControls');\n    node.start();\n    //\n    // // Make the transform control visible so that the user can manipulate the control selection.\n    myControl.transformControls.visible = visible;\n    //\n    // // Attach the model to the transform control\n    myControl.inputs.selection = modelNode;\n    //\n    // // set 'translate' mode to position the selection.\n    myControl.inputs.mode = mode;\n    modelNode.obj3D.controls = myControl; // store gizmoCtrl inside object\n    if(isNewObject){ //i keep the current solution for new objects\n      if (!this.lastObject3D || !this.lastObject3D.controls) {\n        try {\n          modelNode.obj3D.uuid = this.lastObject3D.uuid || this.lastObject3D.id;\n        } catch (e) {\n          console.log(`id obj in Scene was not assigned to id from DB since`);\n        }\n        this.lastObject3D = modelNode.obj3D;\n      }\n    }else{//objects already in place have to become the \"lastObject\" (i think?)\n      console.log(\"in my solution !\");\n      console.log(modelNode);\n      console.log(modelNode.obj3D.uuid);\n      this.lastObject3D = modelNode.obj3D;\n    }\n\n    return modelNode;\n  }\n\n  removeGizmoFromLastObject(): void {\n    this.lastObject3D.controls.transformControls.visible = false;\n    this.lastObject3D.controls.transformControls.dispose();\n    this.lastObject3D.controls = null;\n  }\n\n  toggleViewFrustum(): void {\n    this.securityCameraAnimator.toggleViewFrustum();\n  }\n\n /*public set3DObjectMode(modelNode: any, mode?: string) {\n   if (mode) {\n     this.objectControl.inputs.selection = modelNode;\n     this.objectControl.inputs.visible = true;\n     this.objectControl.inputs.mode = mode;\n\n     setInterval(() => {\n       console.log(modelNode);\n     }, 5000);\n\n   } else {\n     this.objectControl.inputs.visible = true;\n   }\n }*/\n}\n"]}
|
|
1
|
+
import { Inject, Injectable } from '@angular/core';
|
|
2
|
+
import { Subject } from 'rxjs';
|
|
3
|
+
//import * as serialijse from "serialijse";
|
|
4
|
+
import { getDistanceBetweenTwoPoints, poiTypeToString, wait, } from '../helpers.service';
|
|
5
|
+
import { MattertagData } from '../mattertagData';
|
|
6
|
+
import { FeatureType, MattertagActionMode, PoiType, SpModule, TagAction, ViewerInteractions, } from '../types.service';
|
|
7
|
+
import { Object3D } from 'three';
|
|
8
|
+
import { CameraMode } from '../types.service';
|
|
9
|
+
import { NestThermostat } from '../matterport-extensions/nest-thermostat/NestThermostat';
|
|
10
|
+
import { PlaneRenderer } from '../matterport-extensions/nest-thermostat/PlaneRenderer';
|
|
11
|
+
import { CanvasRenderer } from "../matterport-extensions/nest-thermostat/CanvasRenderer";
|
|
12
|
+
import { TvPlayer } from "../matterport-extensions/tv-player/TvPlayer";
|
|
13
|
+
import * as i0 from "@angular/core";
|
|
14
|
+
import * as i1 from "@angular/router";
|
|
15
|
+
import * as i2 from "./baseVisibility.service";
|
|
16
|
+
import * as i3 from "../config";
|
|
17
|
+
export class MatterportService {
|
|
18
|
+
constructor(config, router, activeRoute, visibilityService, ngZone) {
|
|
19
|
+
this.router = router;
|
|
20
|
+
this.activeRoute = activeRoute;
|
|
21
|
+
this.visibilityService = visibilityService;
|
|
22
|
+
this.ngZone = ngZone;
|
|
23
|
+
this.slots = []; //SlotNode[] = [];
|
|
24
|
+
this.nodes = [];
|
|
25
|
+
this.lastCameraPosition = { x: 0.0, y: 0.0, z: 0.0 };
|
|
26
|
+
this.cursorPositionButtonDisplayed = false;
|
|
27
|
+
// Measure mode
|
|
28
|
+
this.isMeasureModeOn = false;
|
|
29
|
+
this.interactionMode = ViewerInteractions.DEFAULT;
|
|
30
|
+
// List of created Mattertag IDs in the current viewer session
|
|
31
|
+
this.mattertagIDs = [];
|
|
32
|
+
// Dictionnary of MattertagID and its data (mattertagData)
|
|
33
|
+
this.dictionnaryTags = new Map();
|
|
34
|
+
this.dictionnaryObjects3D = new Map();
|
|
35
|
+
this.dictionnarySceneObjects3D = new Map();
|
|
36
|
+
this.lastMeasure = [];
|
|
37
|
+
this.distancesLastMeasure = [];
|
|
38
|
+
this.resolution = {
|
|
39
|
+
width: 500,
|
|
40
|
+
height: 600,
|
|
41
|
+
};
|
|
42
|
+
this.visibility = {
|
|
43
|
+
mattertags: false,
|
|
44
|
+
sweeps: true,
|
|
45
|
+
};
|
|
46
|
+
this.tagsAttachments = {};
|
|
47
|
+
this.currentSweep = new Subject();
|
|
48
|
+
//camera position with rotation
|
|
49
|
+
this.currentCameraPose = new Subject();
|
|
50
|
+
this.forbiddenSweeps = [];
|
|
51
|
+
this.tagAction = new Subject();
|
|
52
|
+
this.inTransitionMode = false;
|
|
53
|
+
this.inTransitionSweep = false;
|
|
54
|
+
this.noLightForObjects = true;
|
|
55
|
+
this.currentCameraMode = CameraMode.OUTSIDE;
|
|
56
|
+
this.onCameraModeChanged = new Subject();
|
|
57
|
+
this.onGoToTag = new Subject();
|
|
58
|
+
this.tagMessengerOn = false;
|
|
59
|
+
/**
|
|
60
|
+
* Actions on left click when positioning mattertag in visit
|
|
61
|
+
*/
|
|
62
|
+
this.pointerLeftClickHandler = () => {
|
|
63
|
+
if (this.mattertagToFollow) {
|
|
64
|
+
const mattertagData = this.dictionnaryTags.get(this.mattertagToFollow);
|
|
65
|
+
mattertagData.setPosition({ ...this.poseMatterport.position }); // copy!! not the reference
|
|
66
|
+
mattertagData.setNormal({ ...this.poseMatterport.normal }); // copy!! not the reference
|
|
67
|
+
this.dictionnaryTags.set(this.mattertagToFollow, mattertagData);
|
|
68
|
+
this.updateMatterTagContentForTagID(this.mattertagToFollow);
|
|
69
|
+
}
|
|
70
|
+
this.onValidatedMattertag();
|
|
71
|
+
};
|
|
72
|
+
this.pointerRightClickHandler = (e) => {
|
|
73
|
+
e.preventDefault();
|
|
74
|
+
this.cancelFollowingCursor();
|
|
75
|
+
alert('action cancelled');
|
|
76
|
+
};
|
|
77
|
+
this.pointerMiddleClickHandler = (e) => {
|
|
78
|
+
this.textDisplayCursorPositionPanel.innerHTML = `position:
|
|
79
|
+
${this.pointToString(this.poseMatterport.position)}\n
|
|
80
|
+
normal:
|
|
81
|
+
${this.pointToString(this.poseMatterport.normal)}\n
|
|
82
|
+
floorId:
|
|
83
|
+
${this.poseMatterport.floorId}`;
|
|
84
|
+
// this.textDisplayCursorPositionPanel.style.display = 'visible';
|
|
85
|
+
this.getCursorPositionButton.style.display = 'none';
|
|
86
|
+
};
|
|
87
|
+
this.config = config;
|
|
88
|
+
// TODO: only for dev!
|
|
89
|
+
if (!!this.getCursorPositionButton &&
|
|
90
|
+
(document.location.href.indexOf('dev') !== -1 || document.location.href.indexOf('localhost') !== -1)) {
|
|
91
|
+
this.intervalCursorPointerPosition = setInterval(() => {
|
|
92
|
+
if (!this.poseMatterport) {
|
|
93
|
+
return;
|
|
94
|
+
}
|
|
95
|
+
const nextShow = this.poseMatterport.time + 1000;
|
|
96
|
+
if (new Date().getTime() > nextShow) {
|
|
97
|
+
if (this.cursorPositionButtonDisplayed) {
|
|
98
|
+
return;
|
|
99
|
+
}
|
|
100
|
+
const size = {
|
|
101
|
+
w: this.container.clientWidth,
|
|
102
|
+
h: this.container.clientHeight,
|
|
103
|
+
};
|
|
104
|
+
const coord = this.sdk.Conversion.worldToScreen(this.poseMatterport.position, this.poseCamera, size);
|
|
105
|
+
this.getCursorPositionButton.style.left = `${coord.x - 25}px`;
|
|
106
|
+
this.getCursorPositionButton.style.top = `${coord.y - 22}px`;
|
|
107
|
+
this.getCursorPositionButton.style.display = 'block';
|
|
108
|
+
// this.textDisplayCursorPositionPanel.style.display = 'none';
|
|
109
|
+
this.cursorPositionButtonDisplayed = true;
|
|
110
|
+
}
|
|
111
|
+
}, 500);
|
|
112
|
+
}
|
|
113
|
+
}
|
|
114
|
+
get currentSpaceID() {
|
|
115
|
+
return this._currentSpaceID;
|
|
116
|
+
}
|
|
117
|
+
set currentSpaceID(value) {
|
|
118
|
+
this._currentSpaceID = value;
|
|
119
|
+
}
|
|
120
|
+
/**
|
|
121
|
+
* Initializes Matterport and all listeners/data
|
|
122
|
+
* @param tagService BaseTagService (to inject html)
|
|
123
|
+
* @param module SpModule (Museum, Immo) to subscribe only to needed functionnnalities
|
|
124
|
+
* @returns boolean
|
|
125
|
+
*/
|
|
126
|
+
async initSdk(tagService, module = SpModule.IMMO) {
|
|
127
|
+
if (this.sdk) {
|
|
128
|
+
// clean if sdk is running already
|
|
129
|
+
await this.action_delete_all_mattertags();
|
|
130
|
+
}
|
|
131
|
+
this.tagService = tagService;
|
|
132
|
+
this.SPModule = module;
|
|
133
|
+
return new Promise((resolve, reject) => {
|
|
134
|
+
// Retrieve DOM elements
|
|
135
|
+
this.pointerButton = document.querySelector('#viewer-pointer-trick');
|
|
136
|
+
this.container = document.querySelector('#viewer-module');
|
|
137
|
+
const iframe = document.querySelector('#viewer-module');
|
|
138
|
+
if (!iframe) {
|
|
139
|
+
return;
|
|
140
|
+
}
|
|
141
|
+
// Add listeners
|
|
142
|
+
if (this.pointerButton) {
|
|
143
|
+
this.pointerButton.addEventListener('click', this.pointerLeftClickHandler);
|
|
144
|
+
// cancel on right click
|
|
145
|
+
this.pointerButton.addEventListener('contextmenu', this.pointerRightClickHandler);
|
|
146
|
+
}
|
|
147
|
+
// Retrieve DOM elements
|
|
148
|
+
this.getCursorPositionButton = document.querySelector('#button');
|
|
149
|
+
this.textDisplayCursorPositionPanel = document.querySelector('#text');
|
|
150
|
+
if (this.getCursorPositionButton) {
|
|
151
|
+
// get position on Matterport "model" on middle click
|
|
152
|
+
this.getCursorPositionButton.addEventListener('click', this.pointerMiddleClickHandler);
|
|
153
|
+
}
|
|
154
|
+
// Load SDK
|
|
155
|
+
console.log('Loading Matterport SDK');
|
|
156
|
+
const showcaseWindow = iframe.contentWindow;
|
|
157
|
+
iframe.addEventListener('load', async function () {
|
|
158
|
+
try {
|
|
159
|
+
this.sdk = await showcaseWindow.MP_SDK.connect(showcaseWindow, 'qn9wsasuy5h2fzrbrn1nzr0id', '3.5');
|
|
160
|
+
}
|
|
161
|
+
catch (e) {
|
|
162
|
+
console.error(e);
|
|
163
|
+
return;
|
|
164
|
+
}
|
|
165
|
+
// Load Mattertag icons and custom subscriptions
|
|
166
|
+
switch (module) {
|
|
167
|
+
case SpModule.IMMO:
|
|
168
|
+
this.sdk.Asset.registerTexture('icon-ticket', this.config.my_config.icon_ticket);
|
|
169
|
+
this.sdk.Asset.registerTexture('icon-equipment', this.config.my_config.icon_equipment);
|
|
170
|
+
this.sdk.Asset.registerTexture('icon-measure', this.config.my_config.icon_measure);
|
|
171
|
+
this.sdk.Asset.registerTexture('icon-data', this.config.my_config.icon_data);
|
|
172
|
+
this.sdk.Asset.registerTexture('icon-object3d', this.config.my_config.icon_object3d);
|
|
173
|
+
this.sdk.Measurements.data.subscribe({
|
|
174
|
+
onAdded: function (index, item, collection) {
|
|
175
|
+
// console.log(
|
|
176
|
+
// "item added to the collection",
|
|
177
|
+
// index,
|
|
178
|
+
// item,
|
|
179
|
+
// );
|
|
180
|
+
// this.measurements[index] = item;
|
|
181
|
+
this.lastMeasure = item.points;
|
|
182
|
+
}.bind(this),
|
|
183
|
+
onCollectionUpdated: function (collection) {
|
|
184
|
+
// console.log('the entire up-to-date collection', collection);
|
|
185
|
+
this.getDistanceForLastMeasurement();
|
|
186
|
+
}.bind(this),
|
|
187
|
+
});
|
|
188
|
+
this.sdk.Measurements.mode.subscribe(function (measurementModeState) {
|
|
189
|
+
// measurement mode state has changed
|
|
190
|
+
this.isMeasureModeOn = measurementModeState.active;
|
|
191
|
+
// console.log('Is measurement mode currently active? ', measurementModeState.active);
|
|
192
|
+
}.bind(this));
|
|
193
|
+
break;
|
|
194
|
+
case SpModule.MUSEUM:
|
|
195
|
+
this.sdk.Asset.registerTexture('icon-data', this.config.my_config.icon_data);
|
|
196
|
+
break;
|
|
197
|
+
case SpModule.HOTEL:
|
|
198
|
+
this.sdk.Asset.registerTexture('icon_room_available', this.config.my_config.icon_room_available);
|
|
199
|
+
this.sdk.Asset.registerTexture('icon_room_unavailable', this.config.my_config.icon_room_unavailable);
|
|
200
|
+
this.sdk.Asset.registerTexture('icon_room_chosen', this.config.my_config.icon_room_chosen);
|
|
201
|
+
this.sdk.Asset.registerTexture('icon_room_viewed', this.config.my_config.icon_room_viewed);
|
|
202
|
+
break;
|
|
203
|
+
default:
|
|
204
|
+
break;
|
|
205
|
+
}
|
|
206
|
+
this.sdk.Asset.registerTexture('icon-position', this.config.my_config.icon_position);
|
|
207
|
+
// Current Room (used for getting bounding box and eventually lazy load mattertag or object3D inside current Room)
|
|
208
|
+
this.sdk.Room.current.subscribe((currentRoom) => {
|
|
209
|
+
console.log(`hello current Room ${JSON.stringify(currentRoom)}`);
|
|
210
|
+
});
|
|
211
|
+
// 3D position's pointer
|
|
212
|
+
this.sdk.Pointer.intersection.subscribe(function (intersection) {
|
|
213
|
+
this.poseMatterport = intersection;
|
|
214
|
+
this.poseMatterport.time = new Date().getTime();
|
|
215
|
+
if (!this.oldPoseMatterportPosition) {
|
|
216
|
+
this.oldPoseMatterportPosition = {
|
|
217
|
+
x: 0,
|
|
218
|
+
y: 0,
|
|
219
|
+
z: 0,
|
|
220
|
+
};
|
|
221
|
+
}
|
|
222
|
+
if (this.interactionMode !== ViewerInteractions.DEFAULT &&
|
|
223
|
+
this.mattertagToFollow) {
|
|
224
|
+
// follow the pointer and changes the position of the last tag
|
|
225
|
+
// (we are about to validate, but it exists in sdk already)
|
|
226
|
+
this.enable_following_tag(this.mattertagToFollow);
|
|
227
|
+
}
|
|
228
|
+
if (!!this.getCursorPositionButton && !!this.getCursorPositionButton.style &&
|
|
229
|
+
(document.URL.indexOf('https://dev.smarterplan.io') !== -1 || document.location.href.indexOf('localhost') !== -1)) {
|
|
230
|
+
this.getCursorPositionButton.style.display = 'none';
|
|
231
|
+
this.cursorPositionButtonDisplayed = false;
|
|
232
|
+
}
|
|
233
|
+
}.bind(this));
|
|
234
|
+
//Camera mode
|
|
235
|
+
this.sdk.Mode.current.subscribe((mode) => {
|
|
236
|
+
this.inTransitionSweep = false;
|
|
237
|
+
switch (mode) {
|
|
238
|
+
case 'mode.dollhouse':
|
|
239
|
+
this.currentCameraMode = CameraMode.DOLLHOUSE;
|
|
240
|
+
break;
|
|
241
|
+
case 'mode.floorplan':
|
|
242
|
+
this.currentCameraMode = CameraMode.FLOORPLAN;
|
|
243
|
+
break;
|
|
244
|
+
case 'mode.inside':
|
|
245
|
+
this.currentCameraMode = CameraMode.INSIDE;
|
|
246
|
+
break;
|
|
247
|
+
case 'mode.transitioning':
|
|
248
|
+
this.currentCameraMode = CameraMode.TRANSITIONING;
|
|
249
|
+
break;
|
|
250
|
+
default:
|
|
251
|
+
this.currentCameraMode = CameraMode.OUTSIDE;
|
|
252
|
+
}
|
|
253
|
+
this.onCameraModeChanged.next(this.currentCameraMode);
|
|
254
|
+
});
|
|
255
|
+
// Camera's viewpoint
|
|
256
|
+
this.sdk.Camera.pose.subscribe(function subscr(pose) {
|
|
257
|
+
this.poseCamera = pose;
|
|
258
|
+
this.currentCameraPose.next(pose);
|
|
259
|
+
// console.log('Current position is ', pose.position);
|
|
260
|
+
// console.log('Rotation angle is ', pose.rotation);
|
|
261
|
+
// console.log("Sweep UUID is", pose.sweep);
|
|
262
|
+
}.bind(this));
|
|
263
|
+
// subscribe to sweeps
|
|
264
|
+
this.sdk.Sweep.data.subscribe({
|
|
265
|
+
onCollectionUpdated: function subscr(collection) {
|
|
266
|
+
// console.log("Sweep collection updated");
|
|
267
|
+
this.sweeps = Object.keys(collection);
|
|
268
|
+
}.bind(this),
|
|
269
|
+
});
|
|
270
|
+
// subscribe to current sweep
|
|
271
|
+
this.sdk.Sweep.current.subscribe(function subscr(currentSweep) {
|
|
272
|
+
// Change to the current sweep has occurred.
|
|
273
|
+
if (currentSweep.sid === '') {
|
|
274
|
+
// console.log('Not currently stationed at a sweep position');
|
|
275
|
+
}
|
|
276
|
+
else {
|
|
277
|
+
// console.log("emmiting sweep", currentSweep.sid);
|
|
278
|
+
this.currentSweep.next(currentSweep.sid);
|
|
279
|
+
}
|
|
280
|
+
}.bind(this));
|
|
281
|
+
// Subscribe to Floor data
|
|
282
|
+
this.sdk.Floor.data.subscribe({
|
|
283
|
+
onCollectionUpdated: function upd(collection) {
|
|
284
|
+
this.floors = Object.values(collection);
|
|
285
|
+
}.bind(this),
|
|
286
|
+
});
|
|
287
|
+
// on tag click open details page
|
|
288
|
+
this.sdk.on('tag.click', (sid) => {
|
|
289
|
+
// get object of this tag
|
|
290
|
+
try {
|
|
291
|
+
const mattertagData = this.dictionnaryTags.get(sid);
|
|
292
|
+
if (!mattertagData) {
|
|
293
|
+
console.log('no mattertagData to display');
|
|
294
|
+
return;
|
|
295
|
+
}
|
|
296
|
+
const url = tagService.getUrlForSeeDetails(mattertagData.getObject(), mattertagData.getType());
|
|
297
|
+
if (url !== '') {
|
|
298
|
+
this.visibilityService.detailShowing.next(true);
|
|
299
|
+
this.ngZone.run(() => {
|
|
300
|
+
this.router.navigate([url]);
|
|
301
|
+
});
|
|
302
|
+
}
|
|
303
|
+
}
|
|
304
|
+
catch {
|
|
305
|
+
console.log('Cannot show details for tag');
|
|
306
|
+
}
|
|
307
|
+
});
|
|
308
|
+
// Pointer trick
|
|
309
|
+
// Create a div that will appear when the cursor is still
|
|
310
|
+
// It will intercept the click of the mouse before it trigerring the Matterport's tour navigation
|
|
311
|
+
this.timerPointer = setInterval(this.updatePointerTrick.bind(this), 50);
|
|
312
|
+
/**
|
|
313
|
+
* Transitions
|
|
314
|
+
*/
|
|
315
|
+
this.sdk.on('viewmode.changestart', function (to, from) {
|
|
316
|
+
// console.log('Starting to move to ' + to + ' from ' + from);
|
|
317
|
+
this.inTransitionMode = true;
|
|
318
|
+
}.bind(this));
|
|
319
|
+
this.sdk.on('viewmode.changeend', function (oldMode, newMode) {
|
|
320
|
+
// console.log('Ended to move to ' + newMode + ' from ' + oldMode);
|
|
321
|
+
if (newMode !== 'mode.transitioning') {
|
|
322
|
+
this.inTransitionMode = false;
|
|
323
|
+
}
|
|
324
|
+
}.bind(this));
|
|
325
|
+
this.sdk.on(this.sdk.Sweep.Event.ENTER, function (oldSweep, newSweep) {
|
|
326
|
+
// console.log('Leaving sweep ' + oldSweep);
|
|
327
|
+
// console.log('Entering sweep ' + newSweep);
|
|
328
|
+
this.inTransitionSweep = false;
|
|
329
|
+
this.displayAzimutalCrown();
|
|
330
|
+
}.bind(this));
|
|
331
|
+
this.sdk.on(this.sdk.Sweep.Event.EXIT, function (fromSweep, toSweep) {
|
|
332
|
+
// console.log('Leaving sweep ' + fromSweep);
|
|
333
|
+
// console.log('Transitioning to sweep ' + toSweep);
|
|
334
|
+
this.inTransitionSweep = true;
|
|
335
|
+
}.bind(this));
|
|
336
|
+
// TODO: get scene with getter instead!
|
|
337
|
+
const [sceneObject] = await this.sdk.Scene.createObjects(1);
|
|
338
|
+
const node = sceneObject.addNode();
|
|
339
|
+
node.start();
|
|
340
|
+
this.threeJSScene = node.obj3D.parent;
|
|
341
|
+
// await this.sdk.Scene.configure((renderer: any, three: any) => {
|
|
342
|
+
// renderer.physicallyCorrectLights = true;
|
|
343
|
+
// renderer.outputEncoding = three.sRGBEncoding;
|
|
344
|
+
// renderer.shadowMap.enabled = true;
|
|
345
|
+
// renderer.shadowMap.bias = 0.0001;
|
|
346
|
+
// renderer.shadowMap.type = three.PCFSoftShadowMap;
|
|
347
|
+
// });
|
|
348
|
+
// TODO: wait for MP ticket resolution before decomment these!
|
|
349
|
+
// Wait until Showcase is actually playing....
|
|
350
|
+
// this.sdk.Tag.data.subscribe({
|
|
351
|
+
// onAdded: async function (index, item, collection) {
|
|
352
|
+
//
|
|
353
|
+
// let thisOpacity = 0.2;
|
|
354
|
+
// this.sdk.Tag.editOpacity(index, thisOpacity);
|
|
355
|
+
//
|
|
356
|
+
// let source = null;
|
|
357
|
+
// try {
|
|
358
|
+
// source = await Promise.all([
|
|
359
|
+
// this.sdk.Sensor.createSource(this.sdk.Sensor.SourceType.SPHERE, {
|
|
360
|
+
// origin: item.anchorPosition,
|
|
361
|
+
// radius: Number(3),
|
|
362
|
+
// userData: {
|
|
363
|
+
// id: index + '-sphere-source',
|
|
364
|
+
// },
|
|
365
|
+
// })
|
|
366
|
+
// ]);
|
|
367
|
+
// } catch (e) {
|
|
368
|
+
// console.log('could not create Sphere sensor')
|
|
369
|
+
// console.error(e);
|
|
370
|
+
// }
|
|
371
|
+
// if (!source) {
|
|
372
|
+
// return;
|
|
373
|
+
// }
|
|
374
|
+
//
|
|
375
|
+
// let sensor = null;
|
|
376
|
+
// try {
|
|
377
|
+
// sensor = await this.sdk.Sensor.createSensor(this.sdk.Sensor.SensorType.CAMERA);
|
|
378
|
+
// } catch (e) {
|
|
379
|
+
// console.log('could not create Camera sensor')
|
|
380
|
+
// console.error(e);
|
|
381
|
+
// }
|
|
382
|
+
// if (!sensor) {
|
|
383
|
+
// return;
|
|
384
|
+
// }
|
|
385
|
+
//
|
|
386
|
+
// sensor.addSource(...source);
|
|
387
|
+
// sensor.readings.subscribe({
|
|
388
|
+
// onUpdated(source, reading) {
|
|
389
|
+
// console.log(thisOpacity);
|
|
390
|
+
// let oldOpacity = thisOpacity;
|
|
391
|
+
// if (reading.inRange) {
|
|
392
|
+
// thisOpacity = 1;
|
|
393
|
+
// console.log(index + ' is inRange');
|
|
394
|
+
// } else if (reading.inView) {
|
|
395
|
+
// console.log(index + ' is inView but not inRange');
|
|
396
|
+
// thisOpacity = 0.5;
|
|
397
|
+
// } else {
|
|
398
|
+
// thisOpacity = 0.2;
|
|
399
|
+
// console.log(index + ' is not inView or inRange');
|
|
400
|
+
// }
|
|
401
|
+
//
|
|
402
|
+
// this.sdk.Tag.editOpacity(index, thisOpacity);
|
|
403
|
+
// /*
|
|
404
|
+
// let inc = 0.01;
|
|
405
|
+
// if (oldOpacity > thisOpacity) {
|
|
406
|
+
// inc = -0.01;
|
|
407
|
+
// }
|
|
408
|
+
//
|
|
409
|
+
// for(var i = oldOpacity; i != thisOpacity; i=i+inc) {
|
|
410
|
+
// setTimeout(function() {
|
|
411
|
+
// mpSdk.Tag.editOpacity(index, i);
|
|
412
|
+
// console.log('Delay', i);
|
|
413
|
+
// },10);
|
|
414
|
+
// }
|
|
415
|
+
// */
|
|
416
|
+
// }
|
|
417
|
+
// });
|
|
418
|
+
// //sensor.showDebug(true);
|
|
419
|
+
// }.bind(this),
|
|
420
|
+
// onCollectionUpdated: function (collection) {
|
|
421
|
+
// console.log('Collection received. There are ', Object.keys(collection).length, ' Tags in the collection', collection);
|
|
422
|
+
// }
|
|
423
|
+
// });
|
|
424
|
+
resolve(true);
|
|
425
|
+
}.bind(this));
|
|
426
|
+
});
|
|
427
|
+
}
|
|
428
|
+
setLightingOff() {
|
|
429
|
+
this.noLightForObjects = true;
|
|
430
|
+
}
|
|
431
|
+
pointToString(point) {
|
|
432
|
+
var x = point.x.toFixed(3);
|
|
433
|
+
var y = point.y.toFixed(3);
|
|
434
|
+
var z = point.z.toFixed(3);
|
|
435
|
+
return `{ x: ${x}, y: ${y}, z: ${z} }`;
|
|
436
|
+
}
|
|
437
|
+
//
|
|
438
|
+
// ---------- Measurements related ----------
|
|
439
|
+
//
|
|
440
|
+
/**
|
|
441
|
+
* Callback after measurement is performed
|
|
442
|
+
*/
|
|
443
|
+
getDistanceForLastMeasurement() {
|
|
444
|
+
if (this.lastMeasure.length > 0) {
|
|
445
|
+
const numberPoints = this.lastMeasure.length;
|
|
446
|
+
this.distancesLastMeasure = [];
|
|
447
|
+
for (let index = 1; index < numberPoints; index += 1) {
|
|
448
|
+
const distance = getDistanceBetweenTwoPoints(this.lastMeasure[index - 1], this.lastMeasure[index]);
|
|
449
|
+
this.distancesLastMeasure.push(distance);
|
|
450
|
+
}
|
|
451
|
+
this.takeScreenShot().then((res) => {
|
|
452
|
+
this.router.navigate([`visit/${this.currentSpaceID}/add_measurement`]);
|
|
453
|
+
});
|
|
454
|
+
}
|
|
455
|
+
}
|
|
456
|
+
getLastDistances() {
|
|
457
|
+
return this.distancesLastMeasure;
|
|
458
|
+
}
|
|
459
|
+
/**
|
|
460
|
+
* Takes screenshot and saves base64 in lastScreenshotUri
|
|
461
|
+
* @returns Promise
|
|
462
|
+
*/
|
|
463
|
+
takeScreenShot() {
|
|
464
|
+
return this.sdk.Renderer.takeScreenShot(this.resolution, this.visibility).then(function (screenShotUri) {
|
|
465
|
+
// base64 string
|
|
466
|
+
this.lastScreenshotUri = screenShotUri;
|
|
467
|
+
return Promise.resolve();
|
|
468
|
+
}.bind(this));
|
|
469
|
+
}
|
|
470
|
+
getScreenShotUri() {
|
|
471
|
+
return this.lastScreenshotUri;
|
|
472
|
+
}
|
|
473
|
+
getLastMeasurement() {
|
|
474
|
+
const data = {
|
|
475
|
+
measure: this.lastMeasure,
|
|
476
|
+
sweep: this.poseCamera.sweep,
|
|
477
|
+
};
|
|
478
|
+
return data;
|
|
479
|
+
}
|
|
480
|
+
//
|
|
481
|
+
// ---------- Utils ----------
|
|
482
|
+
//
|
|
483
|
+
/**
|
|
484
|
+
* Styling of pointer
|
|
485
|
+
*/
|
|
486
|
+
updatePointerTrick() {
|
|
487
|
+
if (this.interactionMode !== ViewerInteractions.DEFAULT &&
|
|
488
|
+
this.mattertagToFollow &&
|
|
489
|
+
this.poseMatterport &&
|
|
490
|
+
this.getDistPosition(this.poseMatterport.position, this.oldPoseMatterportPosition) > 25) {
|
|
491
|
+
this.pointerButton.style.display = 'none';
|
|
492
|
+
const size = {
|
|
493
|
+
w: this.container.clientWidth,
|
|
494
|
+
h: this.container.clientHeight,
|
|
495
|
+
};
|
|
496
|
+
const coords = this.sdk.Conversion.worldToScreen(this.poseMatterport.position, this.poseCamera, size);
|
|
497
|
+
this.pointerButton.style.left = `${coords.x * Math.sign(coords.x) - 25}px`;
|
|
498
|
+
this.pointerButton.style.top = `${coords.y * Math.sign(coords.x) - 25}px`;
|
|
499
|
+
this.oldPoseMatterportPosition = {
|
|
500
|
+
...this.poseMatterport.position,
|
|
501
|
+
};
|
|
502
|
+
}
|
|
503
|
+
else {
|
|
504
|
+
this.pointerButton.style.display = 'block';
|
|
505
|
+
}
|
|
506
|
+
}
|
|
507
|
+
/**
|
|
508
|
+
* Realtime mattertag following the cursor
|
|
509
|
+
* @param mattertag string
|
|
510
|
+
*/
|
|
511
|
+
enable_following_tag(mattertag) {
|
|
512
|
+
this.sdk.Tag.editPosition(mattertag, {
|
|
513
|
+
anchorPosition: {
|
|
514
|
+
x: this.poseMatterport.position.x * 1,
|
|
515
|
+
y: this.poseMatterport.position.y * 1,
|
|
516
|
+
z: this.poseMatterport.position.z * 1,
|
|
517
|
+
},
|
|
518
|
+
stemVector: {
|
|
519
|
+
x: this.poseMatterport.normal.x * 0.3,
|
|
520
|
+
y: this.poseMatterport.normal.y * 0.3,
|
|
521
|
+
z: this.poseMatterport.normal.z * 0.3,
|
|
522
|
+
},
|
|
523
|
+
});
|
|
524
|
+
}
|
|
525
|
+
/**
|
|
526
|
+
* Get the distance betwween two 3D positions
|
|
527
|
+
* Used in order to see how much the cursor has moved from the previous position
|
|
528
|
+
* @param pos1
|
|
529
|
+
* @param pos2
|
|
530
|
+
*/
|
|
531
|
+
getDistPosition(pos1, pos2) {
|
|
532
|
+
const size = {
|
|
533
|
+
w: this.container.clientWidth,
|
|
534
|
+
h: this.container.clientHeight,
|
|
535
|
+
};
|
|
536
|
+
const coords1 = this.sdk.Conversion.worldToScreen(pos1, this.poseCamera, size);
|
|
537
|
+
const coords2 = this.sdk.Conversion.worldToScreen(pos2, this.poseCamera, size);
|
|
538
|
+
return Math.sqrt((coords1.x - coords2.x) ** 2 + (coords1.y - coords2.y) ** 2);
|
|
539
|
+
}
|
|
540
|
+
//
|
|
541
|
+
// ---------- Mattertag related ----------
|
|
542
|
+
//
|
|
543
|
+
/**
|
|
544
|
+
* Creates the Mattertag that will follow the cursor
|
|
545
|
+
* @param mattertagData MattertagData
|
|
546
|
+
*/
|
|
547
|
+
async addCursorMattertag(mattertagData) {
|
|
548
|
+
if (!this.poseMatterport)
|
|
549
|
+
return;
|
|
550
|
+
this.mattertagToFollow = await this.addMattertagToViewer(mattertagData);
|
|
551
|
+
console.log('following the tag', this.mattertagToFollow);
|
|
552
|
+
this.sdk.Tag.editIcon(this.mattertagToFollow, 'icon-position');
|
|
553
|
+
this.sdk.Tag.editOpacity(this.mattertagToFollow, 1);
|
|
554
|
+
}
|
|
555
|
+
/**
|
|
556
|
+
* Adds Mattertag to viewer for an existing object with coordinates (in mattertagData.poi)
|
|
557
|
+
* (position, injected html, set icon)
|
|
558
|
+
* @param mattertagData
|
|
559
|
+
* returns mattertagID
|
|
560
|
+
*/
|
|
561
|
+
async addMattertagToViewer(mattertagData) {
|
|
562
|
+
let sidList;
|
|
563
|
+
if (!this.sdk) {
|
|
564
|
+
return null;
|
|
565
|
+
}
|
|
566
|
+
try {
|
|
567
|
+
sidList = await this.sdk.Tag.add(mattertagData.getData());
|
|
568
|
+
}
|
|
569
|
+
catch (error) {
|
|
570
|
+
console.log('Tag does not belong to the visit', error, mattertagData.getData());
|
|
571
|
+
}
|
|
572
|
+
if (sidList) {
|
|
573
|
+
const mattertagID = sidList[0];
|
|
574
|
+
// console.log("added tag", mattertagData.getType(), mattertagID);
|
|
575
|
+
this.mattertagIDs.push(mattertagID);
|
|
576
|
+
this.dictionnaryTags.set(mattertagID, mattertagData);
|
|
577
|
+
return mattertagID;
|
|
578
|
+
}
|
|
579
|
+
return null;
|
|
580
|
+
}
|
|
581
|
+
/**
|
|
582
|
+
* Actions when position of mattertag is validated by left click
|
|
583
|
+
*/
|
|
584
|
+
onValidatedMattertag() {
|
|
585
|
+
this.setInteractionMode(ViewerInteractions.DEFAULT);
|
|
586
|
+
const lastTag = this.getLastTag();
|
|
587
|
+
if (lastTag) {
|
|
588
|
+
const callbackMode = this.dictionnaryTags
|
|
589
|
+
.get(lastTag)
|
|
590
|
+
.getCallbackActionMode();
|
|
591
|
+
this.callbackAfterMattertagValidation(callbackMode);
|
|
592
|
+
}
|
|
593
|
+
}
|
|
594
|
+
/**
|
|
595
|
+
* Registers new icon (path to image) and set its for Mattertag
|
|
596
|
+
* @param mattertagID string
|
|
597
|
+
* @param iconPath string
|
|
598
|
+
*/
|
|
599
|
+
async addNewIconAndSetForTag(mattertagID, iconPath) {
|
|
600
|
+
await this.sdk.Asset.registerTexture(`icon_${mattertagID}`, iconPath);
|
|
601
|
+
this.sdk.Tag.editIcon(mattertagID, `icon_${mattertagID}`);
|
|
602
|
+
}
|
|
603
|
+
/**
|
|
604
|
+
* Changes icon of Mattertag (the iconName should be registered = one of default ones)
|
|
605
|
+
* @param mattertagID string
|
|
606
|
+
* @param iconName string
|
|
607
|
+
*/
|
|
608
|
+
async setRegistredIconForTag(mattertagID, iconName) {
|
|
609
|
+
try {
|
|
610
|
+
this.sdk.Tag.editIcon(mattertagID, iconName);
|
|
611
|
+
}
|
|
612
|
+
catch (e) {
|
|
613
|
+
console.log('could not edit Icon with name ', iconName, '. Is it registered?');
|
|
614
|
+
}
|
|
615
|
+
}
|
|
616
|
+
/**
|
|
617
|
+
* Sets default icon for a tag (registered in initSdk) OR uses tagIcon from POI (available from MattertagData)
|
|
618
|
+
* @param mattertagID string
|
|
619
|
+
* @param mattertagData MattertagData
|
|
620
|
+
* @returns
|
|
621
|
+
*/
|
|
622
|
+
async setTagIconAndOpacity(mattertagID, mattertagData) {
|
|
623
|
+
if (this.SPModule === SpModule.HOTEL) {
|
|
624
|
+
const room = mattertagData.getObject();
|
|
625
|
+
let iconName = `icon_room`;
|
|
626
|
+
if (room.available) {
|
|
627
|
+
iconName = `${iconName}_available`;
|
|
628
|
+
}
|
|
629
|
+
else {
|
|
630
|
+
iconName = `${iconName}_unavailable`;
|
|
631
|
+
}
|
|
632
|
+
this.sdk.Tag.editIcon(mattertagID, iconName);
|
|
633
|
+
return;
|
|
634
|
+
}
|
|
635
|
+
const stringTagType = poiTypeToString(mattertagData.getType());
|
|
636
|
+
let opacity = this.config.my_config.DEFAULT_OPACITY_TAG;
|
|
637
|
+
let iconName = `icon-${stringTagType}`;
|
|
638
|
+
const poi = mattertagData.getPoi();
|
|
639
|
+
if (poi && poi.tagIcon) {
|
|
640
|
+
const tagIcon = JSON.parse(poi.tagIcon);
|
|
641
|
+
if (mattertagData.getType() === PoiType.DATA) {
|
|
642
|
+
const feature = mattertagData.getObject();
|
|
643
|
+
if (feature.type === FeatureType.INDICATOR_TEMP) {
|
|
644
|
+
tagIcon.src = this.tagService.getIconTagImageForFeature(feature, poi);
|
|
645
|
+
}
|
|
646
|
+
}
|
|
647
|
+
if (tagIcon.src) {
|
|
648
|
+
const pathSigned = await this.tagService.getSignedTagIconSource(tagIcon.src);
|
|
649
|
+
if (pathSigned) {
|
|
650
|
+
try {
|
|
651
|
+
iconName = `icon-${stringTagType}-${mattertagID}-${mattertagData.customIconIndex}`;
|
|
652
|
+
await this.sdk.Asset.registerTexture(iconName, pathSigned);
|
|
653
|
+
mattertagData.customIconIndex += 1;
|
|
654
|
+
}
|
|
655
|
+
catch {
|
|
656
|
+
console.log('error registerIcon');
|
|
657
|
+
}
|
|
658
|
+
}
|
|
659
|
+
}
|
|
660
|
+
if (tagIcon.opacity) {
|
|
661
|
+
opacity = tagIcon.opacity;
|
|
662
|
+
}
|
|
663
|
+
}
|
|
664
|
+
try {
|
|
665
|
+
this.sdk.Tag.editIcon(mattertagID, iconName);
|
|
666
|
+
this.sdk.Tag.editOpacity(mattertagID, opacity);
|
|
667
|
+
}
|
|
668
|
+
catch (error) {
|
|
669
|
+
console.log('Error editIcon or opacity', error);
|
|
670
|
+
}
|
|
671
|
+
}
|
|
672
|
+
/**
|
|
673
|
+
* Moves viewer to last tag created
|
|
674
|
+
*/
|
|
675
|
+
goToLastTag() {
|
|
676
|
+
setTimeout(() => {
|
|
677
|
+
const lastTag = this.getLastTag();
|
|
678
|
+
this.goToTag(lastTag);
|
|
679
|
+
}, 2000);
|
|
680
|
+
}
|
|
681
|
+
/**
|
|
682
|
+
* Moves viewer to Mattertag with ID provided
|
|
683
|
+
* @param mattertagID string
|
|
684
|
+
* @returns
|
|
685
|
+
*/
|
|
686
|
+
async goToTag(mattertagID) {
|
|
687
|
+
if (mattertagID === '')
|
|
688
|
+
return;
|
|
689
|
+
try {
|
|
690
|
+
this.onGoToTag.next(mattertagID);
|
|
691
|
+
await this.sdk.Sweep.current.waitUntil((currentSweep) => currentSweep !== '');
|
|
692
|
+
await this.sdk.Mattertag.navigateToTag(mattertagID, this.sdk.Mattertag.Transition.FLY);
|
|
693
|
+
}
|
|
694
|
+
catch (error) {
|
|
695
|
+
console.log('cannot navigate to tag', error);
|
|
696
|
+
}
|
|
697
|
+
}
|
|
698
|
+
/**
|
|
699
|
+
* Updates content of Mattertag with mattertagID (billboard, injected html, tag icon)
|
|
700
|
+
* @param mattertagID string
|
|
701
|
+
* @param object Ticket, Equipment, Feature, etc
|
|
702
|
+
* @param tagType PoiType
|
|
703
|
+
*/
|
|
704
|
+
async updateMatterTagContentForTagID(mattertagID, object = null, tagType = null) {
|
|
705
|
+
this.sdk.Tag.editBillboard(mattertagID, this.dictionnaryTags.get(mattertagID).getData());
|
|
706
|
+
if (object && tagType) {
|
|
707
|
+
await this.injectHtmlInTag(tagType, object, mattertagID);
|
|
708
|
+
}
|
|
709
|
+
await this.setTagIconAndOpacity(mattertagID, this.dictionnaryTags.get(mattertagID));
|
|
710
|
+
}
|
|
711
|
+
/**
|
|
712
|
+
* Updates injected html for Mattertag
|
|
713
|
+
* @param mattertagID string
|
|
714
|
+
* @param object Ticket, Equipment, Feature, etc
|
|
715
|
+
* @param tagType PoiType
|
|
716
|
+
*/
|
|
717
|
+
async updateInjectedHtmlForTagID(mattertagID, object, tagType) {
|
|
718
|
+
await this.injectHtmlInTag(tagType, object, mattertagID);
|
|
719
|
+
}
|
|
720
|
+
/**
|
|
721
|
+
* Deletes Mattertag from Viewer by its ID
|
|
722
|
+
* @param mattertagID string
|
|
723
|
+
*/
|
|
724
|
+
deleteMattertagFromId(mattertagID) {
|
|
725
|
+
this.sdk.Tag.remove(mattertagID);
|
|
726
|
+
this.dictionnaryTags.delete(mattertagID);
|
|
727
|
+
}
|
|
728
|
+
/**
|
|
729
|
+
* Deletes latest created mattertag
|
|
730
|
+
*/
|
|
731
|
+
deleteLastMattertag() {
|
|
732
|
+
const mattertagID = this.mattertagIDs.pop();
|
|
733
|
+
if (mattertagID) {
|
|
734
|
+
this.deleteMattertagFromId(mattertagID);
|
|
735
|
+
}
|
|
736
|
+
}
|
|
737
|
+
/**
|
|
738
|
+
* Legacy: used to be called action_add_mattertag_from_POI
|
|
739
|
+
* Adds and configures Mattertag for an object (Ticket, Equipment, Feature, etc) that corresponds to POI (coordinates, tagIcon)
|
|
740
|
+
* @param tagType PoiType
|
|
741
|
+
* @param object Ticket, Equipment, Feature...
|
|
742
|
+
* @param poi POI
|
|
743
|
+
* @returns
|
|
744
|
+
*/
|
|
745
|
+
async createMattertagFromPOI(tagType, object, poi) {
|
|
746
|
+
// check if tag exists already
|
|
747
|
+
const { tag, sweep } = this.getTagFromElementId(object.id);
|
|
748
|
+
if (tag) {
|
|
749
|
+
// console.log("tag exists", object)
|
|
750
|
+
return;
|
|
751
|
+
}
|
|
752
|
+
const mattertagData = new MattertagData(tagType);
|
|
753
|
+
mattertagData.setObject(object, tagType);
|
|
754
|
+
if (poi.coordinate) {
|
|
755
|
+
mattertagData.setPosition(JSON.parse(poi.coordinate));
|
|
756
|
+
}
|
|
757
|
+
if (poi.metadata) {
|
|
758
|
+
const tagMetadata = JSON.parse(poi.metadata);
|
|
759
|
+
if (tagMetadata.normal) {
|
|
760
|
+
mattertagData.setNormal(tagMetadata.normal);
|
|
761
|
+
}
|
|
762
|
+
else {
|
|
763
|
+
mattertagData.setNormal({ x: 0, y: -0.15, z: 0 });
|
|
764
|
+
}
|
|
765
|
+
}
|
|
766
|
+
if (poi.matterportSweepID) {
|
|
767
|
+
mattertagData.setSweepID(poi.matterportSweepID);
|
|
768
|
+
}
|
|
769
|
+
mattertagData.setPoi(poi);
|
|
770
|
+
const createdTagID = await this.addMattertagToViewer(mattertagData);
|
|
771
|
+
if (createdTagID && this.sdk) {
|
|
772
|
+
await this.setTagIconAndOpacity(createdTagID, mattertagData);
|
|
773
|
+
await this.injectHtmlInTag(tagType, object, createdTagID);
|
|
774
|
+
}
|
|
775
|
+
}
|
|
776
|
+
/**
|
|
777
|
+
* Inject custom HTML as Mattertag content
|
|
778
|
+
* @param tagType PoiType
|
|
779
|
+
* @param object Ticket, Equipment, Feature etc
|
|
780
|
+
* @param tagID string
|
|
781
|
+
*/
|
|
782
|
+
async injectHtmlInTag(tagType, object, tagID) {
|
|
783
|
+
let html = await this.tagService.getHtmlToInject(tagType, object);
|
|
784
|
+
if (html !== '' && this.sdk) {
|
|
785
|
+
const scriptTag = this.tagService.getScriptForTag(object, tagType);
|
|
786
|
+
html += `${scriptTag}`;
|
|
787
|
+
// create and register the sandbox
|
|
788
|
+
const [sandboxId, messenger] = await this.sdk.Tag.registerSandbox(html);
|
|
789
|
+
// detach previous sandbox from a tag
|
|
790
|
+
let attachmentID = this.tagsAttachments[tagID];
|
|
791
|
+
if (attachmentID) {
|
|
792
|
+
this.sdk.Tag.detach(tagID, attachmentID);
|
|
793
|
+
}
|
|
794
|
+
this.tagsAttachments[tagID] = sandboxId;
|
|
795
|
+
// attach the sandbox to a tag
|
|
796
|
+
this.sdk.Tag.attach(tagID, sandboxId);
|
|
797
|
+
// receive data from the sandbox
|
|
798
|
+
// tagMessengerOn allows to go here only once
|
|
799
|
+
if (!this.tagMessengerOn) {
|
|
800
|
+
this.tagMessengerOn = true;
|
|
801
|
+
const imageClick = (featureID) => {
|
|
802
|
+
// console.log("image click handler", featureID);
|
|
803
|
+
this.tagService.onActionImageClick(featureID);
|
|
804
|
+
};
|
|
805
|
+
const audioClick = (audioCommentID) => {
|
|
806
|
+
// console.log("audio click handler", audioCommentID);
|
|
807
|
+
this.tagService.onActionAudioClick(audioCommentID);
|
|
808
|
+
};
|
|
809
|
+
const videoClick = (url) => {
|
|
810
|
+
this.tagService.onActionVideoClick(url);
|
|
811
|
+
};
|
|
812
|
+
messenger.on(TagAction.DETAIL_CLICK, this.tagService.onActionDetailClick.bind(this.tagService));
|
|
813
|
+
messenger.on(TagAction.TICKET_CLICK, this.tagService.onActionDetailClick.bind(this.tagService));
|
|
814
|
+
messenger.on(TagAction.AUDIO_CLICK, audioClick);
|
|
815
|
+
messenger.on(TagAction.IMAGE_CLICK, imageClick);
|
|
816
|
+
messenger.on(TagAction.VIDEO_CLICK, videoClick);
|
|
817
|
+
messenger.on(TagAction.DOC_CLICK, this.tagService.onActionDocClick.bind(this.tagService));
|
|
818
|
+
messenger.on(TagAction.YOUTUBE_CLICK, this.tagService.onActionYoutubeClick.bind(this.tagService));
|
|
819
|
+
}
|
|
820
|
+
}
|
|
821
|
+
else {
|
|
822
|
+
// if html is empty (case of EMBED content), we edit billboard for Feature and attach new content
|
|
823
|
+
const { comment, tagDescription } = this.tagService.getBillboardMediaToEmbed(object);
|
|
824
|
+
if (comment) {
|
|
825
|
+
// register the media
|
|
826
|
+
const [attachmentID] = await this.sdk.Tag.registerAttachment(comment.externalLink);
|
|
827
|
+
// attach
|
|
828
|
+
this.sdk.Tag.attach(tagID, attachmentID);
|
|
829
|
+
this.sdk.Tag.editBillboard(tagID, {
|
|
830
|
+
label: object.title,
|
|
831
|
+
description: tagDescription,
|
|
832
|
+
});
|
|
833
|
+
}
|
|
834
|
+
}
|
|
835
|
+
}
|
|
836
|
+
async action_delete_all_mattertags() {
|
|
837
|
+
await this.sdk.Tag.remove(...this.mattertagIDs);
|
|
838
|
+
this.mattertagIDs = [];
|
|
839
|
+
this.dictionnaryTags.clear();
|
|
840
|
+
}
|
|
841
|
+
/**
|
|
842
|
+
* Deletes Mattertag from visit associated with object ID (ticketID, etc)
|
|
843
|
+
* @param elementID string
|
|
844
|
+
*/
|
|
845
|
+
async deleteMattertagForObject(elementID) {
|
|
846
|
+
const matterTagID = this.getTagFromElementId(elementID).tag;
|
|
847
|
+
if (matterTagID) {
|
|
848
|
+
try {
|
|
849
|
+
await this.sdk.Tag.remove(matterTagID);
|
|
850
|
+
this.dictionnaryTags.delete(matterTagID);
|
|
851
|
+
const index = this.mattertagIDs.indexOf(matterTagID);
|
|
852
|
+
this.mattertagIDs.splice(index, 1);
|
|
853
|
+
}
|
|
854
|
+
catch (error) {
|
|
855
|
+
console.log('Cannot delete tag', matterTagID, error);
|
|
856
|
+
}
|
|
857
|
+
}
|
|
858
|
+
}
|
|
859
|
+
/**
|
|
860
|
+
* uuid from threejs
|
|
861
|
+
* @param uuid
|
|
862
|
+
*/
|
|
863
|
+
async deleteObject3D(uuid) {
|
|
864
|
+
this.dictionnaryObjects3D.get(uuid).obj3D.visible = false;
|
|
865
|
+
}
|
|
866
|
+
getObject3DModelNodeFromDictionnary(uuid) {
|
|
867
|
+
let obj = this.dictionnaryObjects3D.get(uuid);
|
|
868
|
+
if (!obj) {
|
|
869
|
+
console.log("weird thing again");
|
|
870
|
+
return null;
|
|
871
|
+
}
|
|
872
|
+
//might break things or fix everything ..?
|
|
873
|
+
if (obj.obj3D.uuid != uuid) {
|
|
874
|
+
console.log("we have THE problem");
|
|
875
|
+
obj.obj3D.uuid = uuid; //not enugh to fix the pb
|
|
876
|
+
}
|
|
877
|
+
return obj;
|
|
878
|
+
}
|
|
879
|
+
/**
|
|
880
|
+
* Creates MattertagData and start repositioning (creates temporary mattertag that follows the cursor)
|
|
881
|
+
* @param poiType PoiType
|
|
882
|
+
* @param element Ticket, Equipment, Feature, etc
|
|
883
|
+
*/
|
|
884
|
+
async addMattertagWhenRepositioning(poiType, element) {
|
|
885
|
+
const mattertagData = new MattertagData(poiType);
|
|
886
|
+
// set the coordinates of the existing tag
|
|
887
|
+
const [poi] = element.pois.items;
|
|
888
|
+
if (poi && poi.coordinate) {
|
|
889
|
+
mattertagData.setPosition(JSON.parse(poi.coordinate));
|
|
890
|
+
mattertagData.setPoi(poi); // to keep custom tagIcon and opacity
|
|
891
|
+
}
|
|
892
|
+
mattertagData.setSweepID(this.poseCamera.sweep);
|
|
893
|
+
mattertagData.setRotation(this.poseCamera.rotation);
|
|
894
|
+
mattertagData.setObject(element, poiType);
|
|
895
|
+
this.mattertagToFollow = await this.addMattertagToViewer(mattertagData);
|
|
896
|
+
this.setInteractionMode(ViewerInteractions.POSITIONING);
|
|
897
|
+
await this.addCursorMattertag(mattertagData);
|
|
898
|
+
}
|
|
899
|
+
/**
|
|
900
|
+
* Creates MattertagData and mattertag that follows the cursor when choosing position for a new object
|
|
901
|
+
* @param poiType
|
|
902
|
+
*/
|
|
903
|
+
async addMattertagWhenAdding(poiType) {
|
|
904
|
+
const mattertagData = new MattertagData(poiType);
|
|
905
|
+
mattertagData.setSweepID(this.poseCamera.sweep);
|
|
906
|
+
mattertagData.setRotation(this.poseCamera.rotation);
|
|
907
|
+
this.setInteractionMode(ViewerInteractions.ADDING);
|
|
908
|
+
await this.addCursorMattertag(mattertagData);
|
|
909
|
+
}
|
|
910
|
+
/**
|
|
911
|
+
* Cancels following of cursor (meaning deleting last Mattertag)
|
|
912
|
+
*/
|
|
913
|
+
cancelFollowingCursor() {
|
|
914
|
+
this.deleteLastMattertag();
|
|
915
|
+
this.setInteractionMode(ViewerInteractions.DEFAULT);
|
|
916
|
+
}
|
|
917
|
+
setLastObject3D(lastObject3D) {
|
|
918
|
+
this.lastObject3D = lastObject3D;
|
|
919
|
+
}
|
|
920
|
+
getLastObject3D() {
|
|
921
|
+
return this.lastObject3D;
|
|
922
|
+
}
|
|
923
|
+
/**
|
|
924
|
+
* Performs callback after mattertag position was validated (creation of object or repositioning)
|
|
925
|
+
* @param mode MattertagActionMode
|
|
926
|
+
*/
|
|
927
|
+
callbackAfterMattertagValidation(mode) {
|
|
928
|
+
this.visibilityService.detailShowing.next(true);
|
|
929
|
+
const lastTag = this.getLastTag();
|
|
930
|
+
switch (mode) {
|
|
931
|
+
case MattertagActionMode.ADD_EQUIPMENT:
|
|
932
|
+
this.goToLastTag();
|
|
933
|
+
this.router.navigate([`visit/${this.currentSpaceID}/add_equip`]);
|
|
934
|
+
break;
|
|
935
|
+
case MattertagActionMode.ADD_TICKET:
|
|
936
|
+
this.goToLastTag();
|
|
937
|
+
this.router.navigate([`visit/${this.currentSpaceID}/add_ticket`]);
|
|
938
|
+
break;
|
|
939
|
+
case MattertagActionMode.ADD_OBJECT3D:
|
|
940
|
+
this.goToLastTag();
|
|
941
|
+
this.router.navigate([`visit/${this.currentSpaceID}/add_object3d`]);
|
|
942
|
+
break;
|
|
943
|
+
case MattertagActionMode.ADD_DATA:
|
|
944
|
+
this.goToLastTag();
|
|
945
|
+
let url;
|
|
946
|
+
if (this.router.url.includes('?')) {
|
|
947
|
+
url = this.router.url.substring(0, this.router.url.indexOf('?'));
|
|
948
|
+
}
|
|
949
|
+
else {
|
|
950
|
+
url = this.router.url;
|
|
951
|
+
}
|
|
952
|
+
this.router.navigate([`${url}/add_feature`]);
|
|
953
|
+
break;
|
|
954
|
+
case MattertagActionMode.ADD_DESK:
|
|
955
|
+
this.goToLastTag();
|
|
956
|
+
this.router.navigate([`visit/${this.currentSpaceID}/add_feature`], {
|
|
957
|
+
queryParams: { isDesk: true },
|
|
958
|
+
});
|
|
959
|
+
break;
|
|
960
|
+
case MattertagActionMode.ADD_ROOM:
|
|
961
|
+
this.goToLastTag();
|
|
962
|
+
this.router.navigate([`visit/${this.currentSpaceID}/add_room`]);
|
|
963
|
+
break;
|
|
964
|
+
case MattertagActionMode.POSITION_OBJECT3D:
|
|
965
|
+
if (lastTag) {
|
|
966
|
+
// DO Nothing (routing), just
|
|
967
|
+
this.router.navigate([`visit/${this.currentSpaceID}/object3d/${this.lastObject3D.id}`], { queryParams: { positioning: true } });
|
|
968
|
+
}
|
|
969
|
+
break;
|
|
970
|
+
case MattertagActionMode.POSITION_TICKET:
|
|
971
|
+
if (lastTag) {
|
|
972
|
+
this.router.navigate([
|
|
973
|
+
`visit/${this.currentSpaceID}/detail/${this.dictionnaryTags.get(lastTag).elementID}`,
|
|
974
|
+
], { queryParams: { positioning: true } });
|
|
975
|
+
}
|
|
976
|
+
break;
|
|
977
|
+
case MattertagActionMode.POSITION_EQUIPMENT:
|
|
978
|
+
if (lastTag) {
|
|
979
|
+
this.router.navigate([
|
|
980
|
+
`visit/${this.currentSpaceID}/equip/${this.dictionnaryTags.get(lastTag).elementID}`,
|
|
981
|
+
], { queryParams: { positioning: true } });
|
|
982
|
+
}
|
|
983
|
+
break;
|
|
984
|
+
case MattertagActionMode.POSITION_DATA:
|
|
985
|
+
if (lastTag) {
|
|
986
|
+
this.router.navigate([], {
|
|
987
|
+
relativeTo: this.activeRoute,
|
|
988
|
+
queryParams: { positioning: true }, queryParamsHandling: "merge"
|
|
989
|
+
});
|
|
990
|
+
}
|
|
991
|
+
break;
|
|
992
|
+
case MattertagActionMode.POSITION_ROOM:
|
|
993
|
+
if (lastTag) {
|
|
994
|
+
this.router.navigate([], {
|
|
995
|
+
relativeTo: this.activeRoute,
|
|
996
|
+
queryParams: { positioning: true }, queryParamsHandling: "merge"
|
|
997
|
+
});
|
|
998
|
+
}
|
|
999
|
+
break;
|
|
1000
|
+
}
|
|
1001
|
+
}
|
|
1002
|
+
/**
|
|
1003
|
+
* Fully updates existing Mattertag content with data of object (Ticket, Equipment, Desk)
|
|
1004
|
+
* @param mattertagID string
|
|
1005
|
+
* @param object Ticket, Equipment, Feature, Desk
|
|
1006
|
+
* @param poiType PoiType
|
|
1007
|
+
* @param poi POI
|
|
1008
|
+
*/
|
|
1009
|
+
async setObjectAndPoiInTag(mattertagID, object, poiType, poi = null) {
|
|
1010
|
+
this.dictionnaryTags.get(mattertagID).setObject(object, poiType);
|
|
1011
|
+
if (poi) {
|
|
1012
|
+
this.dictionnaryTags.get(mattertagID).setPoi(poi);
|
|
1013
|
+
}
|
|
1014
|
+
this.dictionnaryTags.get(mattertagID).elementID = object.id;
|
|
1015
|
+
try {
|
|
1016
|
+
await this.updateMatterTagContentForTagID(mattertagID, object, poiType);
|
|
1017
|
+
// TODO: fix this
|
|
1018
|
+
await this.updateMatterTagPosInSdkViewer(mattertagID, object, poiType, poi);
|
|
1019
|
+
}
|
|
1020
|
+
catch (e) {
|
|
1021
|
+
console.log(`error in setObjectAndPoiInTag: ${e}`);
|
|
1022
|
+
}
|
|
1023
|
+
}
|
|
1024
|
+
async updateMatterTagPosInSdkViewer(mattertagID, object, poiType, poi = null) {
|
|
1025
|
+
const IndexToUpdate = object.pois.items.findIndex((u) => u.id === poi.id);
|
|
1026
|
+
object.pois.items[IndexToUpdate] = poi;
|
|
1027
|
+
const { x, y, z } = JSON.parse(poi.coordinate);
|
|
1028
|
+
const newPosition = {
|
|
1029
|
+
anchorPosition: {
|
|
1030
|
+
x: x,
|
|
1031
|
+
y: y,
|
|
1032
|
+
z: z,
|
|
1033
|
+
},
|
|
1034
|
+
stemVector: {
|
|
1035
|
+
// make the Tag stick straight up and make it 0.30 meters (~1 foot) tall
|
|
1036
|
+
x: 0,
|
|
1037
|
+
y: 0.3,
|
|
1038
|
+
z: 0,
|
|
1039
|
+
},
|
|
1040
|
+
};
|
|
1041
|
+
this.sdk.Mattertag.editPosition(mattertagID, newPosition);
|
|
1042
|
+
this.sdk.Tag.editPosition(mattertagID, newPosition);
|
|
1043
|
+
this.dictionnaryTags
|
|
1044
|
+
.get(mattertagID)
|
|
1045
|
+
.setPosition(newPosition.anchorPosition);
|
|
1046
|
+
for (let tagId of this.mattertagIDs) {
|
|
1047
|
+
const currentTag = this.dictionnaryTags.get(tagId);
|
|
1048
|
+
if (!currentTag) {
|
|
1049
|
+
continue;
|
|
1050
|
+
}
|
|
1051
|
+
if (currentTag.elementID === object.id && tagId !== mattertagID) {
|
|
1052
|
+
this.dictionnaryTags.delete(tagId);
|
|
1053
|
+
this.sdk.Tag.remove(tagId);
|
|
1054
|
+
this.sdk.Mattertag.remove(tagId);
|
|
1055
|
+
break;
|
|
1056
|
+
}
|
|
1057
|
+
}
|
|
1058
|
+
}
|
|
1059
|
+
/**
|
|
1060
|
+
* Gets matterTagID and its sweep for an object ID (ticket ID, etc)
|
|
1061
|
+
* @param elementID string
|
|
1062
|
+
* @returns {tag: string | null, sweep: string | null}
|
|
1063
|
+
*/
|
|
1064
|
+
getTagFromElementId(elementID) {
|
|
1065
|
+
let tagID = null;
|
|
1066
|
+
let sweepID = null;
|
|
1067
|
+
for (let [mattertagID, mattertagData] of this.dictionnaryTags) {
|
|
1068
|
+
if (mattertagData.elementID === elementID) {
|
|
1069
|
+
tagID = mattertagID;
|
|
1070
|
+
const sweep = mattertagData.getSweepID();
|
|
1071
|
+
if (sweep && this.sweeps && this.sweeps.includes(sweep)) {
|
|
1072
|
+
sweepID = sweep;
|
|
1073
|
+
}
|
|
1074
|
+
}
|
|
1075
|
+
}
|
|
1076
|
+
return { tag: tagID, sweep: sweepID };
|
|
1077
|
+
}
|
|
1078
|
+
/**
|
|
1079
|
+
* Gets latest tag created in the visit (when following the cursor to position)
|
|
1080
|
+
* @returns string mattertagID
|
|
1081
|
+
*/
|
|
1082
|
+
getLastTag() {
|
|
1083
|
+
if (this.mattertagToFollow) {
|
|
1084
|
+
return this.mattertagToFollow;
|
|
1085
|
+
}
|
|
1086
|
+
else {
|
|
1087
|
+
return this.mattertagIDs.length === 0
|
|
1088
|
+
? null
|
|
1089
|
+
: this.mattertagIDs[this.mattertagIDs.length - 1];
|
|
1090
|
+
}
|
|
1091
|
+
}
|
|
1092
|
+
/**
|
|
1093
|
+
* Gets MattertagData for Mattertag (replaces getTagDataFromId)
|
|
1094
|
+
* @param mattertagID
|
|
1095
|
+
* @returns MattertagData
|
|
1096
|
+
*/
|
|
1097
|
+
getMatterTagDataForMattertag(mattertagID) {
|
|
1098
|
+
try {
|
|
1099
|
+
return this.dictionnaryTags.get(mattertagID);
|
|
1100
|
+
}
|
|
1101
|
+
catch {
|
|
1102
|
+
console.log('cannot retrieve mattertagData for tag', mattertagID);
|
|
1103
|
+
}
|
|
1104
|
+
return null;
|
|
1105
|
+
}
|
|
1106
|
+
//
|
|
1107
|
+
// ---------- Viewer related (switch views, go to) ----------
|
|
1108
|
+
//
|
|
1109
|
+
async action_toolbox_floorplan() {
|
|
1110
|
+
if (this.inTransitionMode || this.inTransitionSweep) {
|
|
1111
|
+
console.log('viewer is in transition, cannot go floorplan');
|
|
1112
|
+
return;
|
|
1113
|
+
}
|
|
1114
|
+
try {
|
|
1115
|
+
await this.sdk.Mode.moveTo('mode.floorplan');
|
|
1116
|
+
}
|
|
1117
|
+
catch (e) {
|
|
1118
|
+
console.log('cannot go to floorplan', e);
|
|
1119
|
+
}
|
|
1120
|
+
}
|
|
1121
|
+
action_toolbox_inside_view() {
|
|
1122
|
+
this.sdk.Mode.moveTo('mode.inside');
|
|
1123
|
+
}
|
|
1124
|
+
actionShowAllFloors() {
|
|
1125
|
+
try {
|
|
1126
|
+
this.sdk.Floor.showAll();
|
|
1127
|
+
}
|
|
1128
|
+
catch (e) {
|
|
1129
|
+
console.log('cannot show all floors', e);
|
|
1130
|
+
}
|
|
1131
|
+
}
|
|
1132
|
+
async action_toolbox_dollhouse() {
|
|
1133
|
+
setTimeout(async () => {
|
|
1134
|
+
// console.log("mode: ", this.inTransitionMode, " sweep: ", this.inTransitionSweep);
|
|
1135
|
+
if (this.inTransitionMode || this.inTransitionSweep) {
|
|
1136
|
+
console.log('viewer is in transition, cannot go dollhouse');
|
|
1137
|
+
return;
|
|
1138
|
+
}
|
|
1139
|
+
try {
|
|
1140
|
+
await this.sdk.Mode.moveTo('mode.dollhouse');
|
|
1141
|
+
}
|
|
1142
|
+
catch (e) {
|
|
1143
|
+
console.log('cannot go to dollhouse', e);
|
|
1144
|
+
}
|
|
1145
|
+
}, 1200);
|
|
1146
|
+
}
|
|
1147
|
+
action_toolbox_mesure() {
|
|
1148
|
+
const newState = !this.isMeasureModeOn;
|
|
1149
|
+
this.sdk.Measurements.toggleMode(newState).then(() => {
|
|
1150
|
+
console.log(`Measurement mode is now ${newState ? 'enabled' : 'disabled'}`);
|
|
1151
|
+
});
|
|
1152
|
+
}
|
|
1153
|
+
action_toolbox_cancel_mesure() {
|
|
1154
|
+
if (this.isMeasureModeOn) {
|
|
1155
|
+
this.action_toolbox_mesure();
|
|
1156
|
+
}
|
|
1157
|
+
}
|
|
1158
|
+
async action_go_to_floor(floorName, matterportFloorSequence = null) {
|
|
1159
|
+
if (!this.floors) {
|
|
1160
|
+
console.log('Floor are not loaded yet');
|
|
1161
|
+
return;
|
|
1162
|
+
}
|
|
1163
|
+
// console.log(this.floors);
|
|
1164
|
+
// look up for sequence number (the safest method)
|
|
1165
|
+
let floorMatterport = this.floors.find((floor) => floor.sequence === matterportFloorSequence);
|
|
1166
|
+
if (!floorMatterport) {
|
|
1167
|
+
floorMatterport = this.floors.find((floor) => floorName.includes(floor.name) && floor.name != '');
|
|
1168
|
+
}
|
|
1169
|
+
if (!floorMatterport) {
|
|
1170
|
+
floorMatterport = this.floors.find((floor) => floorName.includes(floor.id));
|
|
1171
|
+
}
|
|
1172
|
+
// console.log(floorMatterport)
|
|
1173
|
+
if (floorMatterport) {
|
|
1174
|
+
let retry = true;
|
|
1175
|
+
while (retry) {
|
|
1176
|
+
try {
|
|
1177
|
+
const floorIndex = await this.sdk.Floor.moveTo(floorMatterport.sequence);
|
|
1178
|
+
// console.log("moved to floorIndex", floorIndex);
|
|
1179
|
+
retry = false;
|
|
1180
|
+
}
|
|
1181
|
+
catch (error) {
|
|
1182
|
+
console.log('Cannot move to Floor', error);
|
|
1183
|
+
await wait(100);
|
|
1184
|
+
}
|
|
1185
|
+
}
|
|
1186
|
+
}
|
|
1187
|
+
else {
|
|
1188
|
+
console.warn('No matterport floor found to move to');
|
|
1189
|
+
}
|
|
1190
|
+
}
|
|
1191
|
+
async action_go_to_sweep(sweep, rotation = null) {
|
|
1192
|
+
if (this.forbiddenSweeps.includes(sweep)) {
|
|
1193
|
+
console.log('user is not allowed to go to this sweep');
|
|
1194
|
+
return;
|
|
1195
|
+
}
|
|
1196
|
+
// console.log("going to sweep", sweep, "with rotation:", rotation);
|
|
1197
|
+
setTimeout(async () => {
|
|
1198
|
+
if (this.inTransitionMode || this.inTransitionSweep) {
|
|
1199
|
+
console.log('Cannot go to sweep, in transition');
|
|
1200
|
+
return;
|
|
1201
|
+
}
|
|
1202
|
+
try {
|
|
1203
|
+
this.inTransitionSweep = true;
|
|
1204
|
+
await this.sdk.Sweep.moveTo(sweep, {
|
|
1205
|
+
transition: 'transition.instant',
|
|
1206
|
+
transitionTime: 1500,
|
|
1207
|
+
});
|
|
1208
|
+
}
|
|
1209
|
+
catch (error) {
|
|
1210
|
+
this.inTransitionSweep = false;
|
|
1211
|
+
console.log('Cannot move to sweep', error);
|
|
1212
|
+
}
|
|
1213
|
+
if (rotation) {
|
|
1214
|
+
await this.sdk.Camera.setRotation(rotation, { speed: 100 }); // speed is degrees per second
|
|
1215
|
+
}
|
|
1216
|
+
}, 1000);
|
|
1217
|
+
}
|
|
1218
|
+
getCurrentSweep() {
|
|
1219
|
+
if (this.poseCamera) {
|
|
1220
|
+
return this.poseCamera.sweep;
|
|
1221
|
+
}
|
|
1222
|
+
return null;
|
|
1223
|
+
}
|
|
1224
|
+
getCurrentCameraPosition() {
|
|
1225
|
+
if (this.poseCamera) {
|
|
1226
|
+
return this.poseCamera;
|
|
1227
|
+
}
|
|
1228
|
+
return null;
|
|
1229
|
+
}
|
|
1230
|
+
setInteractionMode(interactionMode) {
|
|
1231
|
+
this.interactionMode = interactionMode;
|
|
1232
|
+
}
|
|
1233
|
+
getInteractionMode() {
|
|
1234
|
+
return this.interactionMode;
|
|
1235
|
+
}
|
|
1236
|
+
/**
|
|
1237
|
+
* Clear all variables, deletes all tags (when viewer is remover or model changed=reload needed)
|
|
1238
|
+
*/
|
|
1239
|
+
async clearAll() {
|
|
1240
|
+
console.log('removing viewer');
|
|
1241
|
+
this.action_delete_all_mattertags();
|
|
1242
|
+
this.floors = null;
|
|
1243
|
+
this.sweeps = null;
|
|
1244
|
+
this.sdk = null;
|
|
1245
|
+
clearInterval(this.timerPointer);
|
|
1246
|
+
this.forbiddenSweeps = [];
|
|
1247
|
+
this.tagMessengerOn = false;
|
|
1248
|
+
// cancel subscriptions
|
|
1249
|
+
this.pointerButton.removeEventListener('click', this.pointerLeftClickHandler);
|
|
1250
|
+
this.pointerButton.removeEventListener('contextmenu', this.pointerRightClickHandler);
|
|
1251
|
+
// TODO: only For Admins
|
|
1252
|
+
if (this.getCursorPositionButton) {
|
|
1253
|
+
this.getCursorPositionButton.removeEventListener('auxclick', this.pointerMiddleClickHandler);
|
|
1254
|
+
}
|
|
1255
|
+
// TODO: only for dev!
|
|
1256
|
+
if (!!this.getCursorPositionButton &&
|
|
1257
|
+
(document.location.href.indexOf('dev') !== -1 || document.location.href.indexOf('localhost') !== -1)) {
|
|
1258
|
+
clearInterval(this.intervalCursorPointerPosition);
|
|
1259
|
+
}
|
|
1260
|
+
}
|
|
1261
|
+
async removeForbiddenSweeps(forbiddenSweeps) {
|
|
1262
|
+
this.forbiddenSweeps = [...forbiddenSweeps];
|
|
1263
|
+
let removed = 0;
|
|
1264
|
+
await Promise.all(forbiddenSweeps.map(async (sweep) => {
|
|
1265
|
+
try {
|
|
1266
|
+
await this.sdk.Sweep.disable(sweep);
|
|
1267
|
+
removed += 1;
|
|
1268
|
+
}
|
|
1269
|
+
catch (error) {
|
|
1270
|
+
console.log(error);
|
|
1271
|
+
}
|
|
1272
|
+
}));
|
|
1273
|
+
console.log('removed sweeps:', removed);
|
|
1274
|
+
}
|
|
1275
|
+
//
|
|
1276
|
+
// ---------- 3D objects (SDK Bundle only) ----------
|
|
1277
|
+
//
|
|
1278
|
+
async init3DObjectViewer() {
|
|
1279
|
+
return new Promise(async (resolve) => {
|
|
1280
|
+
var [sceneObject] = await this.sdk.Scene.createObjects(1);
|
|
1281
|
+
var node = sceneObject.addNode();
|
|
1282
|
+
// TODO change this 🤮
|
|
1283
|
+
node.addComponent('mp.lights');
|
|
1284
|
+
//node.addComponent('mp.lights');
|
|
1285
|
+
/*node.addComponent('mp.lights');
|
|
1286
|
+
node.addComponent('mp.lights');*/
|
|
1287
|
+
node.start();
|
|
1288
|
+
const nodeControl = sceneObject.addNode();
|
|
1289
|
+
this.objectControl = nodeControl.addComponent('mp.transformControls');
|
|
1290
|
+
nodeControl.start();
|
|
1291
|
+
//this.add3DObject({},null);
|
|
1292
|
+
resolve();
|
|
1293
|
+
});
|
|
1294
|
+
}
|
|
1295
|
+
async add3DObject(obj, mode) {
|
|
1296
|
+
return new Promise(async (resolve) => {
|
|
1297
|
+
const [sceneObject] = await this.sdk.Scene.createObjects(1);
|
|
1298
|
+
// TODO: improvment, regroup all of these in Dynamical Objects Lib
|
|
1299
|
+
// SecurityCamera
|
|
1300
|
+
// NestThermostat
|
|
1301
|
+
// Video
|
|
1302
|
+
let isAnimatedSecurityCamera = false;
|
|
1303
|
+
let isNestThermostat = false;
|
|
1304
|
+
let isSmarterplanPromotionalVideo = false;
|
|
1305
|
+
let isAzimuthalCrown = false;
|
|
1306
|
+
/**
|
|
1307
|
+
* TODO: refacto with an enum or switch/case
|
|
1308
|
+
*/
|
|
1309
|
+
if (obj.object === "security_camera") {
|
|
1310
|
+
isAnimatedSecurityCamera = true;
|
|
1311
|
+
}
|
|
1312
|
+
if (obj.object === "nest_thermostat") {
|
|
1313
|
+
isNestThermostat = true;
|
|
1314
|
+
}
|
|
1315
|
+
if (obj.object === 'video') {
|
|
1316
|
+
isSmarterplanPromotionalVideo = true;
|
|
1317
|
+
}
|
|
1318
|
+
if (obj.object === 'azimuth') {
|
|
1319
|
+
isAzimuthalCrown = true;
|
|
1320
|
+
}
|
|
1321
|
+
const modelNode = sceneObject.addNode();
|
|
1322
|
+
let component = null;
|
|
1323
|
+
const initial = {
|
|
1324
|
+
url: `/assets/3Dobjects/objects/${obj.object}${obj.format.indexOf('.') === -1 ? '.' + obj.format : obj.format}`,
|
|
1325
|
+
// TODO/ store localPosition && localRotation in BDD too (in order to have pertfect initial placement)
|
|
1326
|
+
localRotation: { "x": 0, "y": 0, "z": 0 },
|
|
1327
|
+
// TODO/ store localPosition && localRotation in BDD too (in order to have pertfect initial placement)
|
|
1328
|
+
localPosition: { "x": 0, "y": 0, "z": 0 },
|
|
1329
|
+
visible: true,
|
|
1330
|
+
colliderEnabled: true
|
|
1331
|
+
};
|
|
1332
|
+
switch (obj.format) {
|
|
1333
|
+
case '.obj':
|
|
1334
|
+
case 'obj':
|
|
1335
|
+
component = modelNode.addComponent('mp.objLoader', initial);
|
|
1336
|
+
break;
|
|
1337
|
+
case '.fbx':
|
|
1338
|
+
case 'fbx':
|
|
1339
|
+
component = modelNode.addComponent('mp.fbxLoader', initial);
|
|
1340
|
+
break;
|
|
1341
|
+
case '.gltf':
|
|
1342
|
+
case 'gltf':
|
|
1343
|
+
component = modelNode.addComponent('mp.gltfLoader', initial);
|
|
1344
|
+
break;
|
|
1345
|
+
case '.glb':
|
|
1346
|
+
case 'glb':
|
|
1347
|
+
component = modelNode.addComponent('mp.gltfLoader', initial);
|
|
1348
|
+
break;
|
|
1349
|
+
case '.dae':
|
|
1350
|
+
case 'dae':
|
|
1351
|
+
component = modelNode.addComponent('mp.daeLoader', initial);
|
|
1352
|
+
break;
|
|
1353
|
+
default:
|
|
1354
|
+
console.log('Format not supported');
|
|
1355
|
+
break;
|
|
1356
|
+
}
|
|
1357
|
+
//cache system (i'll try to make it work later ...)
|
|
1358
|
+
/*let objContentsJSON = JSON.stringify(modelNode);
|
|
1359
|
+
console.log(modelNode);
|
|
1360
|
+
console.log(objContentsJSON);
|
|
1361
|
+
console.log(JSON.stringify(modelNode));
|
|
1362
|
+
localStorage.setItem(`testObj_${obj.object}`, objContentsJSON);
|
|
1363
|
+
console.log("stored ?!");
|
|
1364
|
+
console.log(typeof modelNode);*/
|
|
1365
|
+
//let dataS = serialijse.serialize(modelNode);
|
|
1366
|
+
//console.log(dataS);
|
|
1367
|
+
// Use 3 ?! Ambient lightings
|
|
1368
|
+
if (this.noLightForObjects) {
|
|
1369
|
+
const lightsNode = sceneObject.addNode();
|
|
1370
|
+
lightsNode.addComponent('mp.ambientLight', {
|
|
1371
|
+
intensity: 1,
|
|
1372
|
+
color: { r: 1.0, g: 1.0, b: 1.0 },
|
|
1373
|
+
});
|
|
1374
|
+
this.noLightForObjects = false;
|
|
1375
|
+
}
|
|
1376
|
+
modelNode.obj3D.position.set(obj.position.x, obj.position.y, obj.position.z);
|
|
1377
|
+
modelNode.obj3D.rotation.set(obj.rotation.x, obj.rotation.y, obj.rotation.z);
|
|
1378
|
+
modelNode.obj3D.scale.set(obj.scale.x, obj.scale.y, obj.scale.z);
|
|
1379
|
+
if (isAzimuthalCrown) {
|
|
1380
|
+
/*if(modelNode.obj3D.scale.x >= 1.0){
|
|
1381
|
+
modelNode.obj3D.scale.set(obj.scale.x/5000, obj.scale.y/5000, obj.scale.z/5000);
|
|
1382
|
+
}*/
|
|
1383
|
+
this.azimuthalCrown = modelNode;
|
|
1384
|
+
this.displayAzimutalCrown();
|
|
1385
|
+
}
|
|
1386
|
+
// By defaut, during creation, object has translation gizmo
|
|
1387
|
+
// => User has to click on lateral panel, and save its position after playing with it!
|
|
1388
|
+
if (mode && !isNestThermostat) {
|
|
1389
|
+
this.attachGizmoControlTo3DObject(modelNode, sceneObject, mode, true, true).catch((e) => console.log(e));
|
|
1390
|
+
}
|
|
1391
|
+
if (this.lastObject3D && typeof this.lastObject3D.id === 'string') {
|
|
1392
|
+
// prompt ThreeJS UUID to match our last generated object if new (not from Db)
|
|
1393
|
+
modelNode.obj3D.uuid = this.lastObject3D.id;
|
|
1394
|
+
}
|
|
1395
|
+
else if (obj.id) {
|
|
1396
|
+
modelNode.obj3D.uuid = obj.id;
|
|
1397
|
+
}
|
|
1398
|
+
this.lastObject3D = modelNode.obj3D;
|
|
1399
|
+
// Store this in memory Map dictionnary
|
|
1400
|
+
//console.log("Adding object: ");
|
|
1401
|
+
//console.log(modelNode);
|
|
1402
|
+
this.dictionnaryObjects3D.set(modelNode.obj3D.uuid, modelNode);
|
|
1403
|
+
this.dictionnarySceneObjects3D.set(modelNode.obj3D.uuid, sceneObject);
|
|
1404
|
+
/*this.sdk.Camera.pose.subscribe(
|
|
1405
|
+
function (pose: any) {
|
|
1406
|
+
//console.log("in callback")
|
|
1407
|
+
//console.log(this.lastCameraPosition)
|
|
1408
|
+
//console.log(pose.position)
|
|
1409
|
+
if((pose.position.x == this.lastCameraPosition.x && pose.position.y == this.lastCameraPosition.y && pose.position.z == this.lastCameraPosition.z) || this.inTransitionMode || this.inTransitionSweep){
|
|
1410
|
+
//console.log("returning")
|
|
1411
|
+
return;
|
|
1412
|
+
}
|
|
1413
|
+
console.log("camera pos:",pose.position);
|
|
1414
|
+
this.lastCameraPosition = {...pose.position};
|
|
1415
|
+
if(this.lastObject3D){
|
|
1416
|
+
this.lastObject3D.position.set(pose.position.x+.5,pose.position.y+.5,pose.position.z+.5);
|
|
1417
|
+
}
|
|
1418
|
+
}.bind(this));*/
|
|
1419
|
+
if (isAnimatedSecurityCamera) {
|
|
1420
|
+
const animator = modelNode.addComponent('mp.securityCamera', {
|
|
1421
|
+
"nearPlane": 0.1,
|
|
1422
|
+
"farPlane": 10,
|
|
1423
|
+
"horizontalFOV": 52,
|
|
1424
|
+
"aspect": 1.7777777777777777,
|
|
1425
|
+
"localPosition": {
|
|
1426
|
+
"x": 0.3,
|
|
1427
|
+
"y": 0.18,
|
|
1428
|
+
"z": 0
|
|
1429
|
+
},
|
|
1430
|
+
"localRotation": {
|
|
1431
|
+
"x": -15,
|
|
1432
|
+
"y": -90,
|
|
1433
|
+
"z": 0
|
|
1434
|
+
},
|
|
1435
|
+
"color": 65280,
|
|
1436
|
+
"panPeriod": 5,
|
|
1437
|
+
"panAngle": -45
|
|
1438
|
+
});
|
|
1439
|
+
const modelOutput = sceneObject.addPath({
|
|
1440
|
+
id: 'animated-model',
|
|
1441
|
+
type: this.sdk.Scene.PathType.OUTPUT,
|
|
1442
|
+
node: modelNode,
|
|
1443
|
+
component: animator,
|
|
1444
|
+
property: 'objectRoot'
|
|
1445
|
+
});
|
|
1446
|
+
this.securityCameraAnimator = animator;
|
|
1447
|
+
if (!obj.viewFrustum) {
|
|
1448
|
+
setTimeout(() => {
|
|
1449
|
+
animator.toggleViewFrustum();
|
|
1450
|
+
}, 1000);
|
|
1451
|
+
}
|
|
1452
|
+
}
|
|
1453
|
+
if (isNestThermostat) {
|
|
1454
|
+
// TODO: use bindPath instead using MP sdk classes (see Security Camera example)
|
|
1455
|
+
// for TV uses CanvasImage below!
|
|
1456
|
+
// const ci = new CanvasImage();
|
|
1457
|
+
// ci.onInit();
|
|
1458
|
+
const cv = new CanvasRenderer();
|
|
1459
|
+
cv.onInit();
|
|
1460
|
+
const plane = new PlaneRenderer();
|
|
1461
|
+
plane.outputs = {
|
|
1462
|
+
objectRoot: new Object3D(),
|
|
1463
|
+
collider: new Object3D()
|
|
1464
|
+
};
|
|
1465
|
+
const inputTexture = cv.outputs.texture;
|
|
1466
|
+
plane.setRootScene(this.threeJSScene);
|
|
1467
|
+
plane.onInit(modelNode, inputTexture);
|
|
1468
|
+
const sc = new NestThermostat();
|
|
1469
|
+
sc.setComponent(component, plane, cv);
|
|
1470
|
+
sc.setRootScene(this.threeJSScene);
|
|
1471
|
+
sc.onInit(modelNode, plane, inputTexture);
|
|
1472
|
+
cv.setCanvasNestThermostatPainter(sc);
|
|
1473
|
+
setTimeout(() => {
|
|
1474
|
+
sc.inputs.loadingState = "Loaded";
|
|
1475
|
+
sc.onInputsUpdated();
|
|
1476
|
+
}, 5000);
|
|
1477
|
+
}
|
|
1478
|
+
if (isSmarterplanPromotionalVideo) {
|
|
1479
|
+
// TODO: use bindPath instead using MP sdk classes (see Security Camera example)
|
|
1480
|
+
const sc = new TvPlayer();
|
|
1481
|
+
sc.setComponent(component);
|
|
1482
|
+
sc.onInit(modelNode);
|
|
1483
|
+
setTimeout(() => {
|
|
1484
|
+
sc.inputs.loadingState = "Loaded";
|
|
1485
|
+
sc.onInputsUpdated();
|
|
1486
|
+
}, 5000);
|
|
1487
|
+
}
|
|
1488
|
+
sceneObject.start();
|
|
1489
|
+
resolve(this.lastObject3D);
|
|
1490
|
+
});
|
|
1491
|
+
}
|
|
1492
|
+
toggleObjectVisibility(objectId) {
|
|
1493
|
+
let obj = this.dictionnaryObjects3D.get(objectId);
|
|
1494
|
+
obj.obj3D.visible = !obj.obj3D.visible;
|
|
1495
|
+
}
|
|
1496
|
+
isObjectVisible(objectId) {
|
|
1497
|
+
let obj = this.dictionnaryObjects3D.get(objectId);
|
|
1498
|
+
return obj.obj3D.visible;
|
|
1499
|
+
}
|
|
1500
|
+
async pointCameraTo3DObject(objectId) {
|
|
1501
|
+
let obj = this.dictionnaryObjects3D.get(objectId);
|
|
1502
|
+
//We create a temporary Tag
|
|
1503
|
+
const poiObject = {
|
|
1504
|
+
coordinate: JSON.stringify(obj.obj3D.position),
|
|
1505
|
+
type: PoiType.OBJECT3D,
|
|
1506
|
+
elementID: objectId, //todo: be careful with this
|
|
1507
|
+
};
|
|
1508
|
+
const objectDb = { id: objectId };
|
|
1509
|
+
try {
|
|
1510
|
+
await this.createMattertagFromPOI(PoiType.OBJECT3D, objectDb, poiObject);
|
|
1511
|
+
}
|
|
1512
|
+
catch (err) { }
|
|
1513
|
+
//Not really necessary anymore since the tag will disappear quick
|
|
1514
|
+
//this.sdk.Tag.editOpacity(mattertagID, 0.0);//opacity);
|
|
1515
|
+
//this.sdk.Tag.allowAction(mattertagID, {}); //disables every action
|
|
1516
|
+
let result = this.getTagFromElementId(objectId);
|
|
1517
|
+
await this.goToTag(result.tag);
|
|
1518
|
+
this.deleteLastMattertag();
|
|
1519
|
+
}
|
|
1520
|
+
getSceneNodeFromObject3DId(uuid) {
|
|
1521
|
+
return this.dictionnarySceneObjects3D.get(uuid);
|
|
1522
|
+
}
|
|
1523
|
+
async displayAzimutalCrown() {
|
|
1524
|
+
if (this.azimuthalCrown) {
|
|
1525
|
+
this.azimuthalCrown.obj3D.position.set(this.poseCamera.position.x, this.poseCamera.position.y, this.poseCamera.position.z);
|
|
1526
|
+
}
|
|
1527
|
+
}
|
|
1528
|
+
async attachGizmoControlTo3DObject(modelNode, sceneObject, mode, visible, isNewObject) {
|
|
1529
|
+
// Create a scene node with a transform control component.
|
|
1530
|
+
let node = null;
|
|
1531
|
+
node = sceneObject.addNode();
|
|
1532
|
+
if (!node) {
|
|
1533
|
+
const [sceneObject] = await this.sdk.Scene.createObjects(1);
|
|
1534
|
+
node = sceneObject.addNode();
|
|
1535
|
+
}
|
|
1536
|
+
const myControl = node.addComponent('mp.transformControls');
|
|
1537
|
+
node.start();
|
|
1538
|
+
//
|
|
1539
|
+
// // Make the transform control visible so that the user can manipulate the control selection.
|
|
1540
|
+
myControl.transformControls.visible = visible;
|
|
1541
|
+
//
|
|
1542
|
+
// // Attach the model to the transform control
|
|
1543
|
+
myControl.inputs.selection = modelNode;
|
|
1544
|
+
//
|
|
1545
|
+
// // set 'translate' mode to position the selection.
|
|
1546
|
+
myControl.inputs.mode = mode;
|
|
1547
|
+
modelNode.obj3D.controls = myControl; // store gizmoCtrl inside object
|
|
1548
|
+
if (isNewObject) { //i keep the current solution for new objects
|
|
1549
|
+
if (!this.lastObject3D || !this.lastObject3D.controls) {
|
|
1550
|
+
try {
|
|
1551
|
+
modelNode.obj3D.uuid = this.lastObject3D.uuid || this.lastObject3D.id;
|
|
1552
|
+
}
|
|
1553
|
+
catch (e) {
|
|
1554
|
+
console.log(`id obj in Scene was not assigned to id from DB since`);
|
|
1555
|
+
}
|
|
1556
|
+
this.lastObject3D = modelNode.obj3D;
|
|
1557
|
+
}
|
|
1558
|
+
}
|
|
1559
|
+
else { //objects already in place have to become the "lastObject" (i think?)
|
|
1560
|
+
console.log("in my solution !");
|
|
1561
|
+
console.log(modelNode);
|
|
1562
|
+
console.log(modelNode.obj3D.uuid);
|
|
1563
|
+
this.lastObject3D = modelNode.obj3D;
|
|
1564
|
+
}
|
|
1565
|
+
return modelNode;
|
|
1566
|
+
}
|
|
1567
|
+
removeGizmoFromLastObject() {
|
|
1568
|
+
this.lastObject3D.controls.transformControls.visible = false;
|
|
1569
|
+
this.lastObject3D.controls.transformControls.dispose();
|
|
1570
|
+
this.lastObject3D.controls = null;
|
|
1571
|
+
}
|
|
1572
|
+
toggleViewFrustum() {
|
|
1573
|
+
this.securityCameraAnimator.toggleViewFrustum();
|
|
1574
|
+
}
|
|
1575
|
+
}
|
|
1576
|
+
MatterportService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.2.7", ngImport: i0, type: MatterportService, deps: [{ token: 'config' }, { token: i1.Router }, { token: i1.ActivatedRoute }, { token: i2.BaseVisibilityService }, { token: i0.NgZone }], target: i0.ɵɵFactoryTarget.Injectable });
|
|
1577
|
+
MatterportService.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "13.2.7", ngImport: i0, type: MatterportService, providedIn: 'root' });
|
|
1578
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.2.7", ngImport: i0, type: MatterportService, decorators: [{
|
|
1579
|
+
type: Injectable,
|
|
1580
|
+
args: [{
|
|
1581
|
+
providedIn: 'root',
|
|
1582
|
+
}]
|
|
1583
|
+
}], ctorParameters: function () { return [{ type: i3.Config, decorators: [{
|
|
1584
|
+
type: Inject,
|
|
1585
|
+
args: ['config']
|
|
1586
|
+
}] }, { type: i1.Router }, { type: i1.ActivatedRoute }, { type: i2.BaseVisibilityService }, { type: i0.NgZone }]; } });
|
|
1587
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"matterport.service.js","sourceRoot":"","sources":["../../../../../projects/ngx-smarterplan-core/src/lib/services/matterport.service.ts"],"names":[],"mappings":"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","sourcesContent":["import {Inject, Injectable, NgZone} from '@angular/core';\r\nimport {ActivatedRoute, Router} from '@angular/router';\r\nimport {Subject} from 'rxjs';\r\n//import * as serialijse from \"serialijse\";\r\nimport {\r\n  getDistanceBetweenTwoPoints,\r\n  poiTypeToString,\r\n  wait,\r\n} from '../helpers.service';\r\nimport {MattertagData} from '../mattertagData';\r\nimport {\r\n  DbObjectType,\r\n  FeatureType,\r\n  IObject3D,\r\n  MattertagActionMode,\r\n  POI,\r\n  PoiType,\r\n  SpModule,\r\n  TagAction,\r\n  ViewerInteractions,\r\n} from '../types.service';\r\nimport {Object3D} from 'three';\r\nimport {BaseTagService} from './tag.service';\r\nimport {BaseVisibilityService} from './baseVisibility.service';\r\nimport {Config} from '../config';\r\nimport {CameraMode} from '../types.service';\r\nimport {ISceneNode, ComponentOutput} from \"../matterport-extensions/scene-component/SceneComponent\";\r\nimport {SecurityCamera} from '../matterport-extensions/security-camera/SecurityCamera';\r\nimport {NestThermostat} from '../matterport-extensions/nest-thermostat/NestThermostat';\r\nimport {PlaneRenderer} from '../matterport-extensions/nest-thermostat/PlaneRenderer';\r\nimport {CanvasRenderer} from \"../matterport-extensions/nest-thermostat/CanvasRenderer\";\r\nimport {VideoRenderer} from \"../matterport-extensions/video-renderer/VideoRenderer\";\r\nimport {HlsLoader} from \"../matterport-extensions/hsl-loader/HlsLoader\";\r\nimport {TvPlayer} from \"../matterport-extensions/tv-player/TvPlayer\";\r\n\r\n// import {CanvasImage} from \"../matterport-extensions/nest-thermostat/CanvasImage\";\r\n\r\n/**\r\n * Custom Threejs Object 3D interface\r\n */\r\ninterface ThreeJSObject3D {\r\n  id: string;\r\n  uuid?: string;\r\n  position?: { x: number, y: number, z: number }\r\n  rotation?: { x: number, y: number, z: number }\r\n  scale?: { x: number, y: number, z: number }\r\n}\r\n\r\ndeclare global {\r\n  interface Window {\r\n    MP_SDK: { connect: Function };\r\n  }\r\n}\r\n\r\n@Injectable({\r\n  providedIn: 'root',\r\n})\r\nexport class MatterportService {\r\n\r\n  private slots: any[] = [];//SlotNode[] = [];\r\n\r\n  private nodes: ISceneNode[] = [];\r\n\r\n  public sdk: any;\r\n\r\n  private container: any;\r\n\r\n  // Position camera\r\n  private poseMatterport: any;\r\n\r\n  public poseCamera: { rotation: any; position: any; sweep: any };\r\n  public lastCameraPosition: any = {x:0.0,y:0.0,z:0.0};\r\n\r\n  private azimuthalCrown:any;\r\n  // Pointer trick\r\n  private pointerButton: any;\r\n\r\n  // Display and get current position of cursor (for Admins only)\r\n  private getCursorPositionButton: any;\r\n  private cursorPositionButtonDisplayed = false;\r\n  private intervalCursorPointerPosition: any;\r\n  private textDisplayCursorPositionPanel: any;\r\n\r\n  private oldPoseMatterportPosition: any;\r\n  // Measure mode\r\n  private isMeasureModeOn = false;\r\n\r\n  private interactionMode: number = ViewerInteractions.DEFAULT;\r\n\r\n  // List of created Mattertag IDs in the current viewer session\r\n  private mattertagIDs: Array<string> = [];\r\n\r\n  // Dictionnary of MattertagID and its data (mattertagData)\r\n  private dictionnaryTags: Map<string, MattertagData> = new Map();\r\n\r\n  private dictionnaryObjects3D: Map<string, any> = new Map();\r\n\r\n  private dictionnarySceneObjects3D: Map<string, any> = new Map();\r\n\r\n  public threeJSScene: any; // global root scene (not used now!)\r\n\r\n  private lastMeasure = [];\r\n\r\n  private distancesLastMeasure: number[] = [];\r\n\r\n  private resolution = {\r\n    width: 500,\r\n    height: 600,\r\n  };\r\n\r\n  private visibility = {\r\n    mattertags: false,\r\n    sweeps: true,\r\n  };\r\n\r\n  public tagsAttachments: Object = {};\r\n\r\n  private lastScreenshotUri: any;\r\n\r\n  public sweeps: Array<string> | null;\r\n\r\n  public currentSweep: Subject<string> = new Subject();\r\n\r\n  //camera position with rotation\r\n  public currentCameraPose: Subject<any> = new Subject();\r\n\r\n  private floors?: Array<{ id: string; name: string; sequence: number }> | null;\r\n\r\n  private _currentSpaceID: string;\r\n\r\n  private lastObject3D: any; // Three JS object\r\n\r\n  private currentFloor;\r\n\r\n  public get currentSpaceID(): string {\r\n    return this._currentSpaceID;\r\n  }\r\n\r\n  public set currentSpaceID(value: string) {\r\n    this._currentSpaceID = value;\r\n  }\r\n\r\n  timerPointer: any;\r\n\r\n  forbiddenSweeps: string[] = [];\r\n\r\n  public tagAction: Subject<{ action: string; data: string }> = new Subject();\r\n\r\n  public mattertagToFollow: string | null;\r\n\r\n  tagService: BaseTagService;\r\n\r\n  config: Config;\r\n\r\n  public inTransitionMode = false;\r\n\r\n  public inTransitionSweep = false;\r\n\r\n  private noLightForObjects = true;\r\n\r\n  public currentCameraMode: CameraMode = CameraMode.OUTSIDE;\r\n\r\n  public onCameraModeChanged = new Subject<CameraMode>();\r\n\r\n  public onGoToTag = new Subject<string>();\r\n\r\n  tagMessengerOn = false;\r\n\r\n  SPModule: SpModule;\r\n\r\n  public objectControl: any;\r\n\r\n  private securityCameraAnimator: any;\r\n\r\n  /**\r\n   * Actions on left click when positioning mattertag in visit\r\n   */\r\n  pointerLeftClickHandler = () => {\r\n    if (this.mattertagToFollow) {\r\n      const mattertagData = this.dictionnaryTags.get(this.mattertagToFollow);\r\n      mattertagData.setPosition({...this.poseMatterport.position}); // copy!! not the reference\r\n      mattertagData.setNormal({...this.poseMatterport.normal}); // copy!! not the reference\r\n      this.dictionnaryTags.set(this.mattertagToFollow, mattertagData);\r\n      this.updateMatterTagContentForTagID(this.mattertagToFollow);\r\n    }\r\n    this.onValidatedMattertag();\r\n  };\r\n\r\n  pointerRightClickHandler = (e) => {\r\n    e.preventDefault();\r\n    this.cancelFollowingCursor();\r\n    alert('action cancelled');\r\n  };\r\n\r\n  pointerMiddleClickHandler = (e) => {\r\n    this.textDisplayCursorPositionPanel.innerHTML = `position:\r\n       ${this.pointToString(this.poseMatterport.position)}\\n\r\n       normal:\r\n       ${this.pointToString(this.poseMatterport.normal)}\\n\r\n       floorId:\r\n       ${this.poseMatterport.floorId}`;\r\n    // this.textDisplayCursorPositionPanel.style.display = 'visible';\r\n    this.getCursorPositionButton.style.display = 'none';\r\n  }\r\n\r\n  constructor(\r\n    @Inject('config') config: Config,\r\n    private router: Router,\r\n    private activeRoute: ActivatedRoute,\r\n    private visibilityService: BaseVisibilityService,\r\n    private ngZone: NgZone\r\n  ) {\r\n    this.config = config;\r\n\r\n    // TODO: only for dev!\r\n    if (!!this.getCursorPositionButton &&\r\n      (document.location.href.indexOf('dev') !== -1 || document.location.href.indexOf('localhost') !== -1)) {\r\n      this.intervalCursorPointerPosition = setInterval(() => {\r\n        if (!this.poseMatterport) {\r\n          return;\r\n        }\r\n\r\n        const nextShow = this.poseMatterport.time + 1000;\r\n        if (new Date().getTime() > nextShow) {\r\n          if (this.cursorPositionButtonDisplayed) {\r\n            return;\r\n          }\r\n\r\n          const size = {\r\n            w: this.container.clientWidth,\r\n            h: this.container.clientHeight,\r\n          };\r\n          const coord = this.sdk.Conversion.worldToScreen(this.poseMatterport.position, this.poseCamera, size);\r\n          this.getCursorPositionButton.style.left = `${coord.x - 25}px`;\r\n          this.getCursorPositionButton.style.top = `${coord.y - 22}px`;\r\n          this.getCursorPositionButton.style.display = 'block';\r\n          // this.textDisplayCursorPositionPanel.style.display = 'none';\r\n          this.cursorPositionButtonDisplayed = true;\r\n        }\r\n      }, 500);\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Initializes Matterport and all listeners/data\r\n   * @param tagService BaseTagService (to inject html)\r\n   * @param module SpModule (Museum, Immo) to subscribe only to needed functionnnalities\r\n   * @returns boolean\r\n   */\r\n  async initSdk(\r\n    tagService: BaseTagService,\r\n    module: SpModule = SpModule.IMMO\r\n  ): Promise<boolean> {\r\n    if (this.sdk) {\r\n      // clean if sdk is running already\r\n      await this.action_delete_all_mattertags();\r\n    }\r\n    this.tagService = tagService;\r\n    this.SPModule = module;\r\n    return new Promise((resolve, reject) => {\r\n      // Retrieve DOM elements\r\n      this.pointerButton = document.querySelector(\r\n        '#viewer-pointer-trick'\r\n      ) as HTMLElement;\r\n      this.container = document.querySelector('#viewer-module') as HTMLElement;\r\n      const iframe = document.querySelector(\r\n        '#viewer-module'\r\n      ) as HTMLIFrameElement;\r\n      if (!iframe) {\r\n        return;\r\n      }\r\n\r\n      // Add listeners\r\n      if (this.pointerButton) {\r\n        this.pointerButton.addEventListener(\r\n          'click',\r\n          this.pointerLeftClickHandler\r\n        );\r\n        // cancel on right click\r\n        this.pointerButton.addEventListener(\r\n          'contextmenu',\r\n          this.pointerRightClickHandler\r\n        );\r\n      }\r\n      // Retrieve DOM elements\r\n      this.getCursorPositionButton = document.querySelector(\r\n        '#button'\r\n      ) as HTMLElement;\r\n      this.textDisplayCursorPositionPanel = document.querySelector(\r\n        '#text'\r\n      ) as HTMLElement;\r\n      if (this.getCursorPositionButton) {\r\n        // get position on Matterport \"model\" on middle click\r\n        this.getCursorPositionButton.addEventListener(\r\n          'click',\r\n          this.pointerMiddleClickHandler\r\n        );\r\n      }\r\n\r\n      // Load SDK\r\n      console.log('Loading Matterport SDK');\r\n      const showcaseWindow = iframe.contentWindow;\r\n      iframe.addEventListener(\r\n        'load',\r\n        async function () {\r\n          try {\r\n            this.sdk = await showcaseWindow.MP_SDK.connect(\r\n              showcaseWindow,\r\n              'qn9wsasuy5h2fzrbrn1nzr0id',\r\n              '3.5'\r\n            );\r\n          } catch (e) {\r\n            console.error(e);\r\n            return;\r\n          }\r\n\r\n          // Load Mattertag icons and custom subscriptions\r\n          switch (module) {\r\n            case SpModule.IMMO:\r\n              this.sdk.Asset.registerTexture(\r\n                'icon-ticket',\r\n                this.config.my_config.icon_ticket\r\n              );\r\n              this.sdk.Asset.registerTexture(\r\n                'icon-equipment',\r\n                this.config.my_config.icon_equipment\r\n              );\r\n              this.sdk.Asset.registerTexture(\r\n                'icon-measure',\r\n                this.config.my_config.icon_measure\r\n              );\r\n              this.sdk.Asset.registerTexture(\r\n                'icon-data',\r\n                this.config.my_config.icon_data\r\n              );\r\n              this.sdk.Asset.registerTexture(\r\n                'icon-object3d',\r\n                this.config.my_config.icon_object3d\r\n              );\r\n\r\n              this.sdk.Measurements.data.subscribe({\r\n                onAdded: function (\r\n                  index: any,\r\n                  item: { points: any },\r\n                  collection: any\r\n                ) {\r\n                  // console.log(\r\n                  //     \"item added to the collection\",\r\n                  //     index,\r\n                  //     item,\r\n                  // );\r\n                  // this.measurements[index] = item;\r\n                  this.lastMeasure = item.points;\r\n                }.bind(this),\r\n                onCollectionUpdated: function (collection: any) {\r\n                  // console.log('the entire up-to-date collection', collection);\r\n                  this.getDistanceForLastMeasurement();\r\n                }.bind(this),\r\n              });\r\n              this.sdk.Measurements.mode.subscribe(\r\n                function (measurementModeState: { active: any }) {\r\n                  // measurement mode state has changed\r\n                  this.isMeasureModeOn = measurementModeState.active;\r\n                  // console.log('Is measurement mode currently active? ', measurementModeState.active);\r\n                }.bind(this)\r\n              );\r\n              break;\r\n            case SpModule.MUSEUM:\r\n              this.sdk.Asset.registerTexture(\r\n                'icon-data',\r\n                this.config.my_config.icon_data\r\n              );\r\n              break;\r\n            case SpModule.HOTEL:\r\n              this.sdk.Asset.registerTexture(\r\n                'icon_room_available',\r\n                this.config.my_config.icon_room_available\r\n              );\r\n              this.sdk.Asset.registerTexture(\r\n                'icon_room_unavailable',\r\n                this.config.my_config.icon_room_unavailable\r\n              );\r\n              this.sdk.Asset.registerTexture(\r\n                'icon_room_chosen',\r\n                this.config.my_config.icon_room_chosen\r\n              );\r\n              this.sdk.Asset.registerTexture(\r\n                'icon_room_viewed',\r\n                this.config.my_config.icon_room_viewed\r\n              );\r\n              break;\r\n            default:\r\n              break;\r\n          }\r\n\r\n          this.sdk.Asset.registerTexture(\r\n            'icon-position',\r\n            this.config.my_config.icon_position\r\n          );\r\n\r\n          // Current Room (used for getting bounding box and eventually lazy load mattertag or object3D inside current Room)\r\n          this.sdk.Room.current.subscribe((currentRoom) => {\r\n            console.log(`hello current Room ${JSON.stringify(currentRoom)}`);\r\n          });\r\n\r\n          // 3D position's pointer\r\n          this.sdk.Pointer.intersection.subscribe(\r\n            function (intersection: any) {\r\n              this.poseMatterport = intersection;\r\n              this.poseMatterport.time = new Date().getTime();\r\n              if (!this.oldPoseMatterportPosition) {\r\n                this.oldPoseMatterportPosition = {\r\n                  x: 0,\r\n                  y: 0,\r\n                  z: 0,\r\n                };\r\n              }\r\n\r\n              if (\r\n                this.interactionMode !== ViewerInteractions.DEFAULT &&\r\n                this.mattertagToFollow\r\n              ) {\r\n                // follow the pointer and changes the position of the last tag\r\n                // (we are about to validate, but it exists in sdk already)\r\n                this.enable_following_tag(this.mattertagToFollow);\r\n              }\r\n              if (!!this.getCursorPositionButton && !!this.getCursorPositionButton.style &&\r\n              (document.URL.indexOf('https://dev.smarterplan.io') !== -1 || document.location.href.indexOf('localhost') !== -1)) {\r\n                this.getCursorPositionButton.style.display = 'none';\r\n                this.cursorPositionButtonDisplayed = false;\r\n              }\r\n            }.bind(this)\r\n          );\r\n\r\n          //Camera mode\r\n          this.sdk.Mode.current.subscribe((mode) => {\r\n            this.inTransitionSweep = false;\r\n            switch (mode) {\r\n              case 'mode.dollhouse':\r\n                this.currentCameraMode = CameraMode.DOLLHOUSE;\r\n                break;\r\n              case 'mode.floorplan':\r\n                this.currentCameraMode = CameraMode.FLOORPLAN;\r\n                break;\r\n              case 'mode.inside':\r\n                this.currentCameraMode = CameraMode.INSIDE;\r\n                break;\r\n              case 'mode.transitioning':\r\n                this.currentCameraMode = CameraMode.TRANSITIONING;\r\n                break;\r\n              default:\r\n                this.currentCameraMode = CameraMode.OUTSIDE;\r\n            }\r\n            this.onCameraModeChanged.next(this.currentCameraMode);\r\n          });\r\n\r\n          // Camera's viewpoint\r\n          this.sdk.Camera.pose.subscribe(\r\n            function subscr(pose: any) {\r\n              this.poseCamera = pose;\r\n\r\n              this.currentCameraPose.next(pose);\r\n              // console.log('Current position is ', pose.position);\r\n              // console.log('Rotation angle is ', pose.rotation);\r\n              // console.log(\"Sweep UUID is\", pose.sweep);\r\n            }.bind(this)\r\n          );\r\n\r\n          // subscribe to sweeps\r\n          this.sdk.Sweep.data.subscribe({\r\n            onCollectionUpdated: function subscr(collection: {}) {\r\n              // console.log(\"Sweep collection updated\");\r\n              this.sweeps = Object.keys(collection);\r\n            }.bind(this),\r\n          });\r\n\r\n          // subscribe to current sweep\r\n          this.sdk.Sweep.current.subscribe(\r\n            function subscr(currentSweep: { sid: string }) {\r\n              // Change to the current sweep has occurred.\r\n              if (currentSweep.sid === '') {\r\n                // console.log('Not currently stationed at a sweep position');\r\n              } else {\r\n                // console.log(\"emmiting sweep\", currentSweep.sid);\r\n                this.currentSweep.next(currentSweep.sid);\r\n              }\r\n            }.bind(this)\r\n          );\r\n\r\n          // Subscribe to Floor data\r\n          this.sdk.Floor.data.subscribe({\r\n            onCollectionUpdated: function upd(this: any, collection: any) {\r\n              this.floors = Object.values(collection);\r\n            }.bind(this),\r\n          });\r\n\r\n          // on tag click open details page\r\n          this.sdk.on('tag.click', (sid: string) => {\r\n            // get object of this tag\r\n            try {\r\n              const mattertagData = this.dictionnaryTags.get(sid);\r\n              if (!mattertagData){\r\n                console.log('no mattertagData to display');\r\n                return;\r\n              }\r\n              const url = tagService.getUrlForSeeDetails(\r\n                mattertagData.getObject(),\r\n                mattertagData.getType()\r\n              );\r\n              if (url !== '') {\r\n                this.visibilityService.detailShowing.next(true);\r\n                this.ngZone.run(() => {\r\n                  this.router.navigate([url]);\r\n                });\r\n              }\r\n            } catch {\r\n              console.log('Cannot show details for tag');\r\n            }\r\n          });\r\n\r\n          // Pointer trick\r\n          // Create a div that will appear when the cursor is still\r\n          // It will intercept the click of the mouse before it trigerring the Matterport's tour navigation\r\n          this.timerPointer = setInterval(\r\n            this.updatePointerTrick.bind(this),\r\n            50\r\n          );\r\n\r\n          /**\r\n           * Transitions\r\n           */\r\n          this.sdk.on(\r\n            'viewmode.changestart',\r\n            function (to, from) {\r\n              // console.log('Starting to move to ' + to + ' from ' + from);\r\n              this.inTransitionMode = true;\r\n            }.bind(this)\r\n          );\r\n\r\n          this.sdk.on(\r\n            'viewmode.changeend',\r\n            function (oldMode, newMode) {\r\n              // console.log('Ended to move to ' + newMode + ' from ' + oldMode);\r\n              if (newMode !== 'mode.transitioning') {\r\n                this.inTransitionMode = false;\r\n              }\r\n            }.bind(this)\r\n          );\r\n\r\n          this.sdk.on(\r\n            this.sdk.Sweep.Event.ENTER,\r\n            function (oldSweep, newSweep) {\r\n              // console.log('Leaving sweep ' + oldSweep);\r\n              // console.log('Entering sweep ' + newSweep);\r\n              this.inTransitionSweep = false;\r\n              this.displayAzimutalCrown();\r\n            }.bind(this)\r\n          );\r\n          this.sdk.on(\r\n            this.sdk.Sweep.Event.EXIT,\r\n            function (fromSweep, toSweep) {\r\n              //   console.log('Leaving sweep ' + fromSweep);\r\n              //   console.log('Transitioning to sweep ' + toSweep);\r\n              this.inTransitionSweep = true;\r\n            }.bind(this)\r\n          );\r\n\r\n          // TODO: get scene with getter instead!\r\n          const [sceneObject] = await this.sdk.Scene.createObjects(1);\r\n          const node = sceneObject.addNode();\r\n          node.start();\r\n          this.threeJSScene = node.obj3D.parent;\r\n\r\n          // await this.sdk.Scene.configure((renderer: any, three: any) => {\r\n          //   renderer.physicallyCorrectLights = true;\r\n          //   renderer.outputEncoding = three.sRGBEncoding;\r\n          //   renderer.shadowMap.enabled = true;\r\n          //   renderer.shadowMap.bias = 0.0001;\r\n          //   renderer.shadowMap.type = three.PCFSoftShadowMap;\r\n          // });\r\n\r\n\r\n          // TODO: wait for MP ticket resolution before decomment these!\r\n          // Wait until Showcase is actually playing....\r\n          // this.sdk.Tag.data.subscribe({\r\n          //   onAdded: async function (index, item, collection) {\r\n          //\r\n          //     let thisOpacity = 0.2;\r\n          //     this.sdk.Tag.editOpacity(index, thisOpacity);\r\n          //\r\n          //     let source = null;\r\n          //     try {\r\n          //       source = await Promise.all([\r\n          //         this.sdk.Sensor.createSource(this.sdk.Sensor.SourceType.SPHERE, {\r\n          //           origin: item.anchorPosition,\r\n          //           radius: Number(3),\r\n          //           userData: {\r\n          //             id: index + '-sphere-source',\r\n          //           },\r\n          //         })\r\n          //       ]);\r\n          //     } catch (e) {\r\n          //       console.log('could not create Sphere sensor')\r\n          //       console.error(e);\r\n          //     }\r\n          //     if (!source) {\r\n          //       return;\r\n          //     }\r\n          //\r\n          //     let sensor = null;\r\n          //     try {\r\n          //       sensor = await this.sdk.Sensor.createSensor(this.sdk.Sensor.SensorType.CAMERA);\r\n          //     } catch (e) {\r\n          //       console.log('could not create Camera sensor')\r\n          //       console.error(e);\r\n          //     }\r\n          //     if (!sensor) {\r\n          //       return;\r\n          //     }\r\n          //\r\n          //     sensor.addSource(...source);\r\n          //     sensor.readings.subscribe({\r\n          //       onUpdated(source, reading) {\r\n          //         console.log(thisOpacity);\r\n          //         let oldOpacity = thisOpacity;\r\n          //         if (reading.inRange) {\r\n          //           thisOpacity = 1;\r\n          //           console.log(index + ' is inRange');\r\n          //         } else if (reading.inView) {\r\n          //           console.log(index + ' is inView but not inRange');\r\n          //           thisOpacity = 0.5;\r\n          //         } else {\r\n          //           thisOpacity = 0.2;\r\n          //           console.log(index + ' is not inView or inRange');\r\n          //         }\r\n          //\r\n          //         this.sdk.Tag.editOpacity(index, thisOpacity);\r\n          //         /*\r\n          //                   let inc = 0.01;\r\n          //                   if (oldOpacity > thisOpacity) {\r\n          //                     inc = -0.01;\r\n          //                   }\r\n          //\r\n          //                   for(var i = oldOpacity; i != thisOpacity; i=i+inc) {\r\n          //                       setTimeout(function() {\r\n          //                          mpSdk.Tag.editOpacity(index, i);\r\n          //                       console.log('Delay', i);\r\n          //                       },10);\r\n          //                   }\r\n          //         */\r\n          //       }\r\n          //     });\r\n          //     //sensor.showDebug(true);\r\n          //   }.bind(this),\r\n          //   onCollectionUpdated: function (collection) {\r\n          //     console.log('Collection received. There are ', Object.keys(collection).length, ' Tags in the collection', collection);\r\n          //   }\r\n          // });\r\n\r\n          resolve(true);\r\n        }.bind(this)\r\n      );\r\n    });\r\n  }\r\n\r\n\r\n  setLightingOff() {\r\n    this.noLightForObjects = true;\r\n  }\r\n\r\n  pointToString(point: { x: number, y: number, z: number }): string {\r\n    var x = point.x.toFixed(3);\r\n    var y = point.y.toFixed(3);\r\n    var z = point.z.toFixed(3);\r\n\r\n    return `{ x: ${x}, y: ${y}, z: ${z} }`;\r\n  }\r\n\r\n  //\r\n  // ---------- Measurements related ----------\r\n  //\r\n  /**\r\n   * Callback after measurement is performed\r\n   */\r\n  getDistanceForLastMeasurement() {\r\n    if (this.lastMeasure.length > 0) {\r\n      const numberPoints = this.lastMeasure.length;\r\n      this.distancesLastMeasure = [];\r\n      for (let index = 1; index < numberPoints; index += 1) {\r\n        const distance = getDistanceBetweenTwoPoints(\r\n          this.lastMeasure[index - 1],\r\n          this.lastMeasure[index]\r\n        );\r\n        this.distancesLastMeasure.push(distance);\r\n      }\r\n      this.takeScreenShot().then((res) => {\r\n        this.router.navigate([`visit/${this.currentSpaceID}/add_measurement`]);\r\n      });\r\n    }\r\n  }\r\n\r\n  getLastDistances() {\r\n    return this.distancesLastMeasure;\r\n  }\r\n\r\n  /**\r\n   * Takes screenshot and saves base64 in lastScreenshotUri\r\n   * @returns Promise\r\n   */\r\n  takeScreenShot(): Promise<any> {\r\n    return this.sdk.Renderer.takeScreenShot(\r\n      this.resolution,\r\n      this.visibility\r\n    ).then(\r\n      function (screenShotUri: any) {\r\n        // base64 string\r\n        this.lastScreenshotUri = screenShotUri;\r\n        return Promise.resolve();\r\n      }.bind(this)\r\n    );\r\n  }\r\n\r\n  getScreenShotUri() {\r\n    return this.lastScreenshotUri;\r\n  }\r\n\r\n  getLastMeasurement(): Object {\r\n    const data = {\r\n      measure: this.lastMeasure,\r\n      sweep: this.poseCamera.sweep,\r\n    };\r\n    return data;\r\n  }\r\n\r\n  //\r\n  // ---------- Utils ----------\r\n  //\r\n  /**\r\n   * Styling of pointer\r\n   */\r\n  updatePointerTrick() {\r\n    if (\r\n      this.interactionMode !== ViewerInteractions.DEFAULT &&\r\n      this.mattertagToFollow &&\r\n      this.poseMatterport &&\r\n      this.getDistPosition(\r\n        this.poseMatterport.position,\r\n        this.oldPoseMatterportPosition\r\n      ) > 25\r\n    ) {\r\n      this.pointerButton.style.display = 'none';\r\n\r\n      const size = {\r\n        w: this.container.clientWidth,\r\n        h: this.container.clientHeight,\r\n      };\r\n      const coords = this.sdk.Conversion.worldToScreen(\r\n        this.poseMatterport.position,\r\n        this.poseCamera,\r\n        size\r\n      );\r\n      this.pointerButton.style.left = `${\r\n        coords.x * Math.sign(coords.x) - 25\r\n      }px`;\r\n      this.pointerButton.style.top = `${coords.y * Math.sign(coords.x) - 25}px`;\r\n      this.oldPoseMatterportPosition = {\r\n        ...this.poseMatterport.position,\r\n      };\r\n    } else {\r\n      this.pointerButton.style.display = 'block';\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Realtime mattertag following the cursor\r\n   * @param mattertag string\r\n   */\r\n  enable_following_tag(mattertag: string) {\r\n    this.sdk.Tag.editPosition(mattertag, {\r\n      anchorPosition: {\r\n        x: this.poseMatterport.position.x * 1,\r\n        y: this.poseMatterport.position.y * 1,\r\n        z: this.poseMatterport.position.z * 1,\r\n      },\r\n      stemVector: {\r\n        x: this.poseMatterport.normal.x * 0.3,\r\n        y: this.poseMatterport.normal.y * 0.3,\r\n        z: this.poseMatterport.normal.z * 0.3,\r\n      },\r\n    });\r\n  }\r\n\r\n  /**\r\n   * Get the distance betwween two 3D positions\r\n   * Used in order to see how much the cursor has moved from the previous position\r\n   * @param pos1\r\n   * @param pos2\r\n   */\r\n  getDistPosition(pos1: any, pos2: any) {\r\n    const size = {\r\n      w: this.container.clientWidth,\r\n      h: this.container.clientHeight,\r\n    };\r\n    const coords1 = this.sdk.Conversion.worldToScreen(\r\n      pos1,\r\n      this.poseCamera,\r\n      size\r\n    );\r\n    const coords2 = this.sdk.Conversion.worldToScreen(\r\n      pos2,\r\n      this.poseCamera,\r\n      size\r\n    );\r\n    return Math.sqrt(\r\n      (coords1.x - coords2.x) ** 2 + (coords1.y - coords2.y) ** 2\r\n    );\r\n  }\r\n\r\n  //\r\n  // ---------- Mattertag related ----------\r\n  //\r\n  /**\r\n   * Creates the Mattertag that will follow the cursor\r\n   * @param mattertagData MattertagData\r\n   */\r\n  async addCursorMattertag(mattertagData: MattertagData) {\r\n    if (!this.poseMatterport) return;\r\n    this.mattertagToFollow = await this.addMattertagToViewer(mattertagData);\r\n    console.log('following the tag', this.mattertagToFollow);\r\n    this.sdk.Tag.editIcon(this.mattertagToFollow, 'icon-position');\r\n    this.sdk.Tag.editOpacity(this.mattertagToFollow, 1);\r\n  }\r\n\r\n  /**\r\n   * Adds Mattertag to viewer for an existing object with coordinates (in mattertagData.poi)\r\n   * (position, injected html, set icon)\r\n   * @param mattertagData\r\n   * returns mattertagID\r\n   */\r\n  async addMattertagToViewer(\r\n    mattertagData: MattertagData\r\n  ): Promise<string | null> {\r\n    let sidList;\r\n    if (!this.sdk) {\r\n      return null;\r\n    }\r\n    try {\r\n      sidList = await this.sdk.Tag.add(mattertagData.getData());\r\n    } catch (error) {\r\n      console.log(\r\n        'Tag does not belong to the visit',\r\n        error,\r\n        mattertagData.getData()\r\n      );\r\n    }\r\n    if (sidList) {\r\n      const mattertagID = sidList[0];\r\n      // console.log(\"added tag\", mattertagData.getType(), mattertagID);\r\n      this.mattertagIDs.push(mattertagID);\r\n      this.dictionnaryTags.set(mattertagID, mattertagData);\r\n      return mattertagID;\r\n    }\r\n    return null;\r\n  }\r\n\r\n  /**\r\n   * Actions when position of mattertag is validated by left click\r\n   */\r\n  onValidatedMattertag() {\r\n    this.setInteractionMode(ViewerInteractions.DEFAULT);\r\n    const lastTag = this.getLastTag();\r\n    if (lastTag) {\r\n      const callbackMode = this.dictionnaryTags\r\n        .get(lastTag)\r\n        .getCallbackActionMode();\r\n      this.callbackAfterMattertagValidation(callbackMode);\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Registers new icon (path to image) and set its for Mattertag\r\n   * @param mattertagID string\r\n   * @param iconPath string\r\n   */\r\n  async addNewIconAndSetForTag(mattertagID: string, iconPath: string) {\r\n    await this.sdk.Asset.registerTexture(`icon_${mattertagID}`, iconPath);\r\n    this.sdk.Tag.editIcon(mattertagID, `icon_${mattertagID}`);\r\n  }\r\n\r\n  /**\r\n   * Changes icon of Mattertag (the iconName should be registered = one of default ones)\r\n   * @param mattertagID string\r\n   * @param iconName string\r\n   */\r\n  async setRegistredIconForTag(mattertagID: string, iconName: string) {\r\n    try {\r\n      this.sdk.Tag.editIcon(mattertagID, iconName);\r\n    } catch (e) {\r\n      console.log(\r\n        'could not edit Icon with name ',\r\n        iconName,\r\n        '. Is it registered?'\r\n      );\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Sets default icon for a tag (registered in initSdk) OR uses tagIcon from POI (available from MattertagData)\r\n   * @param mattertagID string\r\n   * @param mattertagData MattertagData\r\n   * @returns\r\n   */\r\n  async setTagIconAndOpacity(\r\n    mattertagID: string,\r\n    mattertagData: MattertagData\r\n  ) {\r\n    if (this.SPModule === SpModule.HOTEL) {\r\n      const room = mattertagData.getObject();\r\n      let iconName = `icon_room`;\r\n      if (room.available) {\r\n        iconName = `${iconName}_available`;\r\n      } else {\r\n        iconName = `${iconName}_unavailable`;\r\n      }\r\n      this.sdk.Tag.editIcon(mattertagID, iconName);\r\n      return;\r\n    }\r\n    const stringTagType = poiTypeToString(mattertagData.getType());\r\n    let opacity = this.config.my_config.DEFAULT_OPACITY_TAG;\r\n    let iconName = `icon-${stringTagType}`;\r\n    const poi = mattertagData.getPoi();\r\n    if (poi && poi.tagIcon) {\r\n      const tagIcon = JSON.parse(poi.tagIcon);\r\n      if (mattertagData.getType() === PoiType.DATA) {\r\n        const feature = mattertagData.getObject();\r\n        if (feature.type === FeatureType.INDICATOR_TEMP) {\r\n          tagIcon.src = this.tagService.getIconTagImageForFeature(feature, poi);\r\n        }\r\n      }\r\n      if (tagIcon.src) {\r\n        const pathSigned = await this.tagService.getSignedTagIconSource(\r\n          tagIcon.src\r\n        );\r\n        if (pathSigned) {\r\n          try {\r\n            iconName = `icon-${stringTagType}-${mattertagID}-${mattertagData.customIconIndex}`;\r\n            await this.sdk.Asset.registerTexture(iconName, pathSigned);\r\n            mattertagData.customIconIndex += 1;\r\n          } catch {\r\n            console.log('error registerIcon');\r\n          }\r\n        }\r\n      }\r\n      if (tagIcon.opacity) {\r\n        opacity = tagIcon.opacity;\r\n      }\r\n    }\r\n    try {\r\n      this.sdk.Tag.editIcon(mattertagID, iconName);\r\n      this.sdk.Tag.editOpacity(mattertagID, opacity);\r\n    } catch (error) {\r\n      console.log('Error editIcon or opacity', error);\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Moves viewer to last tag created\r\n   */\r\n  goToLastTag() {\r\n    setTimeout(() => {\r\n      const lastTag = this.getLastTag();\r\n      this.goToTag(lastTag);\r\n    }, 2000);\r\n  }\r\n\r\n  /**\r\n   * Moves viewer to Mattertag with ID provided\r\n   * @param mattertagID string\r\n   * @returns\r\n   */\r\n  async goToTag(mattertagID: string) {\r\n    if (mattertagID === '') return;\r\n    try {\r\n      this.onGoToTag.next(mattertagID);\r\n      await this.sdk.Sweep.current.waitUntil(\r\n        (currentSweep) => currentSweep !== ''\r\n      );\r\n      await this.sdk.Mattertag.navigateToTag(\r\n        mattertagID,\r\n        this.sdk.Mattertag.Transition.FLY\r\n      );\r\n    } catch (error) {\r\n      console.log('cannot navigate to tag', error);\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Updates content of Mattertag with mattertagID (billboard, injected html, tag icon)\r\n   * @param mattertagID string\r\n   * @param object Ticket, Equipment, Feature, etc\r\n   * @param tagType PoiType\r\n   */\r\n  async updateMatterTagContentForTagID(\r\n    mattertagID: string,\r\n    object: DbObjectType = null,\r\n    tagType: PoiType = null\r\n  ) {\r\n    this.sdk.Tag.editBillboard(\r\n      mattertagID,\r\n      this.dictionnaryTags.get(mattertagID).getData()\r\n    );\r\n    if (object && tagType) {\r\n      await this.injectHtmlInTag(tagType, object, mattertagID);\r\n    }\r\n    await this.setTagIconAndOpacity(\r\n      mattertagID,\r\n      this.dictionnaryTags.get(mattertagID)\r\n    );\r\n  }\r\n\r\n  /**\r\n   * Updates injected html for Mattertag\r\n   * @param mattertagID string\r\n   * @param object Ticket, Equipment, Feature, etc\r\n   * @param tagType PoiType\r\n   */\r\n  async updateInjectedHtmlForTagID(\r\n    mattertagID: string,\r\n    object: any,\r\n    tagType: PoiType\r\n  ) {\r\n    await this.injectHtmlInTag(tagType, object, mattertagID);\r\n  }\r\n\r\n  /**\r\n   * Deletes Mattertag from Viewer by its ID\r\n   * @param mattertagID string\r\n   */\r\n  deleteMattertagFromId(mattertagID: string) {\r\n    this.sdk.Tag.remove(mattertagID);\r\n    this.dictionnaryTags.delete(mattertagID);\r\n  }\r\n\r\n  /**\r\n   * Deletes latest created mattertag\r\n   */\r\n  deleteLastMattertag() {\r\n    const mattertagID = this.mattertagIDs.pop();\r\n    if (mattertagID) {\r\n      this.deleteMattertagFromId(mattertagID);\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Legacy: used to be called action_add_mattertag_from_POI\r\n   * Adds and configures Mattertag for an object (Ticket, Equipment, Feature, etc) that corresponds to POI (coordinates, tagIcon)\r\n   * @param tagType PoiType\r\n   * @param object Ticket, Equipment, Feature...\r\n   * @param poi POI\r\n   * @returns\r\n   */\r\n  async createMattertagFromPOI(\r\n    tagType: PoiType,\r\n    object: DbObjectType,\r\n    poi: POI\r\n  ): Promise<any> {\r\n    // check if tag exists already\r\n    const {tag, sweep} = this.getTagFromElementId(object.id);\r\n    if (tag) {\r\n      // console.log(\"tag exists\", object)\r\n      return;\r\n    }\r\n    const mattertagData: MattertagData = new MattertagData(tagType);\r\n    mattertagData.setObject(object, tagType);\r\n    if (poi.coordinate) {\r\n      mattertagData.setPosition(JSON.parse(poi.coordinate));\r\n    }\r\n    if (poi.metadata) {\r\n      const tagMetadata = JSON.parse(poi.metadata);\r\n      if (tagMetadata.normal) {\r\n        mattertagData.setNormal(tagMetadata.normal);\r\n      } else {\r\n        mattertagData.setNormal({x: 0, y: -0.15, z: 0});\r\n      }\r\n    }\r\n    if (poi.matterportSweepID) {\r\n      mattertagData.setSweepID(poi.matterportSweepID);\r\n    }\r\n    mattertagData.setPoi(poi);\r\n\r\n    const createdTagID = await this.addMattertagToViewer(mattertagData);\r\n    if (createdTagID && this.sdk) {\r\n      await this.setTagIconAndOpacity(createdTagID, mattertagData);\r\n      await this.injectHtmlInTag(tagType, object, createdTagID);\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Inject custom HTML as Mattertag content\r\n   * @param tagType PoiType\r\n   * @param object Ticket, Equipment, Feature etc\r\n   * @param tagID string\r\n   */\r\n  async injectHtmlInTag(tagType: PoiType, object: DbObjectType, tagID: string) {\r\n    let html = await this.tagService.getHtmlToInject(tagType, object);\r\n    if (html !== '' && this.sdk) {\r\n      const scriptTag = this.tagService.getScriptForTag(object, tagType);\r\n      html += `${scriptTag}`;\r\n      // create and register the sandbox\r\n      const [sandboxId, messenger] = await this.sdk.Tag.registerSandbox(html);\r\n\r\n      // detach previous sandbox from a tag\r\n      let attachmentID = this.tagsAttachments[tagID];\r\n      if (attachmentID) {\r\n        this.sdk.Tag.detach(tagID, attachmentID);\r\n      }\r\n      this.tagsAttachments[tagID] = sandboxId;\r\n\r\n      // attach the sandbox to a tag\r\n      this.sdk.Tag.attach(tagID, sandboxId);\r\n      // receive data from the sandbox\r\n      // tagMessengerOn allows to go here only once\r\n      if (!this.tagMessengerOn) {\r\n        this.tagMessengerOn = true;\r\n        const imageClick = (featureID: string) => {\r\n          // console.log(\"image click handler\", featureID);\r\n          this.tagService.onActionImageClick(featureID);\r\n        };\r\n\r\n        const audioClick = (audioCommentID: string) => {\r\n          // console.log(\"audio click handler\", audioCommentID);\r\n          this.tagService.onActionAudioClick(audioCommentID);\r\n        };\r\n        const videoClick = (url: string) => {\r\n          this.tagService.onActionVideoClick(url);\r\n        };\r\n        messenger.on(\r\n          TagAction.DETAIL_CLICK,\r\n          this.tagService.onActionDetailClick.bind(this.tagService)\r\n        );\r\n        messenger.on(\r\n          TagAction.TICKET_CLICK,\r\n          this.tagService.onActionDetailClick.bind(this.tagService)\r\n        );\r\n        messenger.on(TagAction.AUDIO_CLICK, audioClick);\r\n        messenger.on(TagAction.IMAGE_CLICK, imageClick);\r\n        messenger.on(TagAction.VIDEO_CLICK, videoClick);\r\n        messenger.on(\r\n          TagAction.DOC_CLICK,\r\n          this.tagService.onActionDocClick.bind(this.tagService)\r\n        );\r\n        messenger.on(\r\n          TagAction.YOUTUBE_CLICK,\r\n          this.tagService.onActionYoutubeClick.bind(this.tagService)\r\n        );\r\n      }\r\n    } else {\r\n      // if html is empty (case of EMBED content), we edit billboard for Feature and attach new content\r\n      const {comment, tagDescription} =\r\n        this.tagService.getBillboardMediaToEmbed(object);\r\n      if (comment) {\r\n        // register the media\r\n        const [attachmentID] = await this.sdk.Tag.registerAttachment(\r\n          comment.externalLink\r\n        );\r\n        // attach\r\n        this.sdk.Tag.attach(tagID, attachmentID);\r\n        this.sdk.Tag.editBillboard(tagID, {\r\n          label: object.title,\r\n          description: tagDescription,\r\n        });\r\n      }\r\n    }\r\n  }\r\n\r\n  async action_delete_all_mattertags() {\r\n    await this.sdk.Tag.remove(...this.mattertagIDs);\r\n    this.mattertagIDs = [];\r\n    this.dictionnaryTags.clear();\r\n  }\r\n\r\n  /**\r\n   * Deletes Mattertag from visit associated with object ID (ticketID, etc)\r\n   * @param elementID string\r\n   */\r\n  async deleteMattertagForObject(elementID: string) {\r\n    const matterTagID = this.getTagFromElementId(elementID).tag;\r\n    if (matterTagID) {\r\n      try {\r\n        await this.sdk.Tag.remove(matterTagID);\r\n        this.dictionnaryTags.delete(matterTagID);\r\n        const index = this.mattertagIDs.indexOf(matterTagID);\r\n        this.mattertagIDs.splice(index, 1);\r\n      } catch (error) {\r\n        console.log('Cannot delete tag', matterTagID, error);\r\n      }\r\n    }\r\n  }\r\n\r\n  /**\r\n   * uuid from threejs\r\n   * @param uuid\r\n   */\r\n  async deleteObject3D(uuid: string): Promise<void> {\r\n    this.dictionnaryObjects3D.get(uuid).obj3D.visible = false;\r\n  }\r\n\r\n  getObject3DModelNodeFromDictionnary(uuid: string): any {\r\n      let obj = this.dictionnaryObjects3D.get(uuid);\r\n      if(!obj){\r\n        console.log(\"weird thing again\");\r\n        return null;\r\n      }\r\n      //might break things or fix everything ..?\r\n      if(obj.obj3D.uuid != uuid){\r\n        console.log(\"we have THE problem\");\r\n        obj.obj3D.uuid = uuid; //not enugh to fix the pb\r\n      }\r\n\r\n      return obj;\r\n  }\r\n\r\n  /**\r\n   * Creates  MattertagData and start repositioning (creates temporary mattertag that follows the cursor)\r\n   * @param poiType PoiType\r\n   * @param element Ticket, Equipment, Feature, etc\r\n   */\r\n  async addMattertagWhenRepositioning(poiType: PoiType, element: DbObjectType) {\r\n    const mattertagData = new MattertagData(poiType);\r\n    // set the coordinates of the existing tag\r\n    const [poi] = element.pois.items;\r\n    if (poi && poi.coordinate) {\r\n      mattertagData.setPosition(JSON.parse(poi.coordinate));\r\n      mattertagData.setPoi(poi); // to keep custom tagIcon and opacity\r\n    }\r\n    mattertagData.setSweepID(this.poseCamera.sweep);\r\n    mattertagData.setRotation(this.poseCamera.rotation);\r\n    mattertagData.setObject(element, poiType);\r\n    this.mattertagToFollow = await this.addMattertagToViewer(mattertagData);\r\n    this.setInteractionMode(ViewerInteractions.POSITIONING);\r\n    await this.addCursorMattertag(mattertagData);\r\n  }\r\n\r\n  /**\r\n   * Creates MattertagData and mattertag that follows the cursor when choosing position for a new object\r\n   * @param poiType\r\n   */\r\n  async addMattertagWhenAdding(poiType: PoiType) {\r\n    const mattertagData = new MattertagData(poiType);\r\n    mattertagData.setSweepID(this.poseCamera.sweep);\r\n    mattertagData.setRotation(this.poseCamera.rotation);\r\n    this.setInteractionMode(ViewerInteractions.ADDING);\r\n    await this.addCursorMattertag(mattertagData);\r\n  }\r\n\r\n  /**\r\n   * Cancels following of cursor (meaning deleting last Mattertag)\r\n   */\r\n  cancelFollowingCursor() {\r\n    this.deleteLastMattertag();\r\n    this.setInteractionMode(ViewerInteractions.DEFAULT);\r\n  }\r\n\r\n  setLastObject3D(lastObject3D: ThreeJSObject3D) {\r\n    this.lastObject3D = lastObject3D;\r\n  }\r\n\r\n  getLastObject3D(): ThreeJSObject3D {\r\n    return this.lastObject3D;\r\n  }\r\n\r\n  /**\r\n   * Performs callback after mattertag position was validated (creation of object or repositioning)\r\n   * @param mode MattertagActionMode\r\n   */\r\n  callbackAfterMattertagValidation(mode: MattertagActionMode) {\r\n    this.visibilityService.detailShowing.next(true);\r\n    const lastTag = this.getLastTag();\r\n    switch (mode) {\r\n      case MattertagActionMode.ADD_EQUIPMENT:\r\n        this.goToLastTag();\r\n        this.router.navigate([`visit/${this.currentSpaceID}/add_equip`]);\r\n        break;\r\n      case MattertagActionMode.ADD_TICKET:\r\n        this.goToLastTag();\r\n        this.router.navigate([`visit/${this.currentSpaceID}/add_ticket`]);\r\n        break;\r\n      case MattertagActionMode.ADD_OBJECT3D:\r\n        this.goToLastTag();\r\n        this.router.navigate([`visit/${this.currentSpaceID}/add_object3d`]);\r\n        break;\r\n      case MattertagActionMode.ADD_DATA:\r\n        this.goToLastTag();\r\n        let url: string;\r\n        if (this.router.url.includes('?')) {\r\n          url = this.router.url.substring(0, this.router.url.indexOf('?'));\r\n        } else {\r\n          url = this.router.url;\r\n        }\r\n        this.router.navigate([`${url}/add_feature`]);\r\n        break;\r\n      case MattertagActionMode.ADD_DESK:\r\n        this.goToLastTag();\r\n        this.router.navigate([`visit/${this.currentSpaceID}/add_feature`], {\r\n          queryParams: {isDesk: true},\r\n        });\r\n        break;\r\n      case MattertagActionMode.ADD_ROOM:\r\n        this.goToLastTag();\r\n        this.router.navigate([`visit/${this.currentSpaceID}/add_room`]);\r\n        break;\r\n      case MattertagActionMode.POSITION_OBJECT3D:\r\n        if (lastTag) {\r\n          // DO Nothing (routing), just\r\n          this.router.navigate(\r\n            [`visit/${this.currentSpaceID}/object3d/${this.lastObject3D.id}`],\r\n            {queryParams: {positioning: true}}\r\n          );\r\n        }\r\n        break;\r\n      case MattertagActionMode.POSITION_TICKET:\r\n        if (lastTag) {\r\n          this.router.navigate(\r\n            [\r\n              `visit/${this.currentSpaceID}/detail/${\r\n                this.dictionnaryTags.get(lastTag).elementID\r\n              }`,\r\n            ],\r\n            {queryParams: {positioning: true}}\r\n          );\r\n        }\r\n        break;\r\n      case MattertagActionMode.POSITION_EQUIPMENT:\r\n        if (lastTag) {\r\n          this.router.navigate(\r\n            [\r\n              `visit/${this.currentSpaceID}/equip/${\r\n                this.dictionnaryTags.get(lastTag).elementID\r\n              }`,\r\n            ],\r\n            {queryParams: {positioning: true}}\r\n          );\r\n        }\r\n        break;\r\n      case MattertagActionMode.POSITION_DATA:\r\n        if (lastTag) {\r\n          this.router.navigate([], {\r\n            relativeTo: this.activeRoute,\r\n            queryParams: {positioning: true}, queryParamsHandling : \"merge\"\r\n          });\r\n        }\r\n        break;\r\n      case MattertagActionMode.POSITION_ROOM:\r\n        if (lastTag) {\r\n          this.router.navigate([], {\r\n            relativeTo: this.activeRoute,\r\n            queryParams: {positioning: true}, queryParamsHandling : \"merge\"\r\n          });\r\n        }\r\n        break;\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Fully updates existing Mattertag content with data of object (Ticket, Equipment, Desk)\r\n   * @param mattertagID string\r\n   * @param object Ticket, Equipment, Feature, Desk\r\n   * @param poiType PoiType\r\n   * @param poi POI\r\n   */\r\n  async setObjectAndPoiInTag(\r\n    mattertagID: string,\r\n    object: DbObjectType,\r\n    poiType: PoiType,\r\n    poi: POI | null = null\r\n  ): Promise<void> {\r\n    this.dictionnaryTags.get(mattertagID).setObject(object, poiType);\r\n    if (poi) {\r\n      this.dictionnaryTags.get(mattertagID).setPoi(poi);\r\n    }\r\n    this.dictionnaryTags.get(mattertagID).elementID = object.id;\r\n    try {\r\n      await this.updateMatterTagContentForTagID(mattertagID, object, poiType);\r\n      // TODO: fix this\r\n      await this.updateMatterTagPosInSdkViewer(\r\n        mattertagID,\r\n        object,\r\n        poiType,\r\n        poi\r\n      );\r\n    } catch (e) {\r\n      console.log(`error in setObjectAndPoiInTag: ${e}`);\r\n    }\r\n  }\r\n\r\n  async updateMatterTagPosInSdkViewer(\r\n    mattertagID: string,\r\n    object: DbObjectType,\r\n    poiType: PoiType,\r\n    poi: POI | null = null\r\n  ): Promise<void> {\r\n    const IndexToUpdate = object.pois.items.findIndex((u) => u.id === poi.id);\r\n    object.pois.items[IndexToUpdate] = poi;\r\n    const {x, y, z} = JSON.parse(poi.coordinate);\r\n    const newPosition = {\r\n      anchorPosition: {\r\n        x: x,\r\n        y: y,\r\n        z: z,\r\n      },\r\n      stemVector: {\r\n        // make the Tag stick straight up and make it 0.30 meters (~1 foot) tall\r\n        x: 0,\r\n        y: 0.3,\r\n        z: 0,\r\n      },\r\n    };\r\n    this.sdk.Mattertag.editPosition(mattertagID, newPosition);\r\n    this.sdk.Tag.editPosition(mattertagID, newPosition);\r\n    this.dictionnaryTags\r\n      .get(mattertagID)\r\n      .setPosition(newPosition.anchorPosition);\r\n\r\n    for (let tagId of this.mattertagIDs) {\r\n      const currentTag = this.dictionnaryTags.get(tagId);\r\n      if (!currentTag) {\r\n        continue;\r\n      }\r\n      if (currentTag.elementID === object.id && tagId !== mattertagID) {\r\n        this.dictionnaryTags.delete(tagId);\r\n        this.sdk.Tag.remove(tagId);\r\n        this.sdk.Mattertag.remove(tagId);\r\n        break;\r\n      }\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Gets matterTagID and its sweep for an object ID (ticket ID, etc)\r\n   * @param elementID string\r\n   * @returns {tag: string | null, sweep: string | null}\r\n   */\r\n  public getTagFromElementId(elementID: string): {\r\n    tag: string | null;\r\n    sweep: string | null;\r\n  } {\r\n    let tagID = null;\r\n    let sweepID = null;\r\n    for (let [mattertagID, mattertagData] of this.dictionnaryTags) {\r\n      if (mattertagData.elementID === elementID) {\r\n        tagID = mattertagID;\r\n        const sweep = mattertagData.getSweepID();\r\n        if (sweep && this.sweeps && this.sweeps.includes(sweep)) {\r\n          sweepID = sweep;\r\n        }\r\n      }\r\n    }\r\n    return {tag: tagID, sweep: sweepID};\r\n  }\r\n\r\n  /**\r\n   * Gets latest tag created in the visit (when following the cursor to position)\r\n   * @returns string mattertagID\r\n   */\r\n  getLastTag(): string | null {\r\n    if (this.mattertagToFollow) {\r\n      return this.mattertagToFollow;\r\n    } else {\r\n      return this.mattertagIDs.length === 0\r\n        ? null\r\n        : this.mattertagIDs[this.mattertagIDs.length - 1];\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Gets MattertagData for Mattertag (replaces getTagDataFromId)\r\n   * @param mattertagID\r\n   * @returns MattertagData\r\n   */\r\n  getMatterTagDataForMattertag(mattertagID: string): MattertagData | null {\r\n    try {\r\n      return this.dictionnaryTags.get(mattertagID);\r\n    } catch {\r\n      console.log('cannot retrieve mattertagData for tag', mattertagID);\r\n    }\r\n    return null;\r\n  }\r\n\r\n  //\r\n  // ---------- Viewer related (switch views, go to) ----------\r\n  //\r\n  async action_toolbox_floorplan() {\r\n    if (this.inTransitionMode || this.inTransitionSweep) {\r\n      console.log('viewer is in transition, cannot go floorplan');\r\n      return;\r\n    }\r\n    try {\r\n      await this.sdk.Mode.moveTo('mode.floorplan');\r\n    } catch (e) {\r\n      console.log('cannot go to floorplan', e);\r\n    }\r\n  }\r\n\r\n  action_toolbox_inside_view() {\r\n    this.sdk.Mode.moveTo('mode.inside');\r\n  }\r\n\r\n  actionShowAllFloors() {\r\n    try {\r\n      this.sdk.Floor.showAll();\r\n    } catch (e) {\r\n      console.log('cannot show all floors', e);\r\n    }\r\n  }\r\n\r\n  async action_toolbox_dollhouse() {\r\n    setTimeout(async () => {\r\n      // console.log(\"mode: \", this.inTransitionMode, \" sweep: \", this.inTransitionSweep);\r\n      if (this.inTransitionMode || this.inTransitionSweep) {\r\n        console.log('viewer is in transition, cannot go dollhouse');\r\n        return;\r\n      }\r\n      try {\r\n        await this.sdk.Mode.moveTo('mode.dollhouse');\r\n      } catch (e) {\r\n        console.log('cannot go to dollhouse', e);\r\n      }\r\n    }, 1200);\r\n  }\r\n\r\n  action_toolbox_mesure() {\r\n    const newState = !this.isMeasureModeOn;\r\n    this.sdk.Measurements.toggleMode(newState).then(() => {\r\n      console.log(\r\n        `Measurement mode is now ${newState ? 'enabled' : 'disabled'}`\r\n      );\r\n    });\r\n  }\r\n\r\n  action_toolbox_cancel_mesure() {\r\n    if (this.isMeasureModeOn) {\r\n      this.action_toolbox_mesure();\r\n    }\r\n  }\r\n\r\n  async action_go_to_floor(\r\n    floorName: string,\r\n    matterportFloorSequence: number = null\r\n  ) {\r\n    if (!this.floors) {\r\n      console.log('Floor are not loaded yet');\r\n      return;\r\n    }\r\n    // console.log(this.floors);\r\n    // look up for sequence number (the safest method)\r\n    let floorMatterport = this.floors.find(\r\n      (floor) => floor.sequence === matterportFloorSequence\r\n    );\r\n    if (!floorMatterport) {\r\n      floorMatterport = this.floors.find(\r\n        (floor) => floorName.includes(floor.name) && floor.name != ''\r\n      );\r\n    }\r\n    if (!floorMatterport) {\r\n      floorMatterport = this.floors.find((floor) =>\r\n        floorName.includes(floor.id)\r\n      );\r\n    }\r\n    // console.log(floorMatterport)\r\n    if (floorMatterport) {\r\n      let retry = true;\r\n      while (retry) {\r\n        try {\r\n          const floorIndex = await this.sdk.Floor.moveTo(\r\n            floorMatterport.sequence\r\n          );\r\n          // console.log(\"moved to floorIndex\", floorIndex);\r\n          retry = false;\r\n        } catch (error) {\r\n          console.log('Cannot move to Floor', error);\r\n          await wait(100);\r\n        }\r\n      }\r\n    } else {\r\n      console.warn('No matterport floor found to move to');\r\n    }\r\n  }\r\n\r\n  async action_go_to_sweep(\r\n    sweep: string,\r\n    rotation: { x: number; y: number } = null\r\n  ) {\r\n    if (this.forbiddenSweeps.includes(sweep)) {\r\n      console.log('user is not allowed to go to this sweep');\r\n      return;\r\n    }\r\n    // console.log(\"going to sweep\", sweep, \"with rotation:\", rotation);\r\n    setTimeout(async () => {\r\n      if (this.inTransitionMode || this.inTransitionSweep) {\r\n        console.log('Cannot go to sweep, in transition');\r\n        return;\r\n      }\r\n      try {\r\n        this.inTransitionSweep = true;\r\n        await this.sdk.Sweep.moveTo(sweep, {\r\n          transition: 'transition.instant',\r\n          transitionTime: 1500,\r\n        });\r\n      } catch (error) {\r\n        this.inTransitionSweep = false;\r\n        console.log('Cannot move to sweep', error);\r\n      }\r\n      if (rotation) {\r\n        await this.sdk.Camera.setRotation(rotation, {speed: 100}); // speed is degrees per second\r\n      }\r\n    }, 1000);\r\n  }\r\n\r\n  getCurrentSweep(): string | null {\r\n    if (this.poseCamera) {\r\n      return this.poseCamera.sweep;\r\n    }\r\n    return null;\r\n  }\r\n\r\n  getCurrentCameraPosition(): {\r\n    rotation: any;\r\n    position: any;\r\n    sweep: any;\r\n  } | null {\r\n    if (this.poseCamera) {\r\n      return this.poseCamera;\r\n    }\r\n    return null;\r\n  }\r\n\r\n  setInteractionMode(interactionMode: number) {\r\n    this.interactionMode = interactionMode;\r\n  }\r\n\r\n  getInteractionMode() {\r\n    return this.interactionMode;\r\n  }\r\n\r\n  /**\r\n   * Clear all variables, deletes all tags (when viewer is remover or model changed=reload needed)\r\n   */\r\n  async clearAll() {\r\n    console.log('removing viewer');\r\n    this.action_delete_all_mattertags();\r\n    this.floors = null;\r\n    this.sweeps = null;\r\n    this.sdk = null;\r\n    clearInterval(this.timerPointer);\r\n    this.forbiddenSweeps = [];\r\n    this.tagMessengerOn = false;\r\n    // cancel subscriptions\r\n    this.pointerButton.removeEventListener(\r\n      'click',\r\n      this.pointerLeftClickHandler\r\n    );\r\n    this.pointerButton.removeEventListener(\r\n      'contextmenu',\r\n      this.pointerRightClickHandler\r\n    );\r\n\r\n    // TODO: only For Admins\r\n    if (this.getCursorPositionButton) {\r\n      this.getCursorPositionButton.removeEventListener(\r\n        'auxclick',\r\n        this.pointerMiddleClickHandler\r\n      );\r\n    }\r\n    // TODO: only for dev!\r\n    if (!!this.getCursorPositionButton &&\r\n      (document.location.href.indexOf('dev') !== -1 || document.location.href.indexOf('localhost') !== -1)) {\r\n      clearInterval(this.intervalCursorPointerPosition);\r\n    }\r\n  }\r\n\r\n  async removeForbiddenSweeps(forbiddenSweeps: string[]) {\r\n    this.forbiddenSweeps = [...forbiddenSweeps];\r\n    let removed = 0;\r\n    await Promise.all(\r\n      forbiddenSweeps.map(async (sweep) => {\r\n        try {\r\n          await this.sdk.Sweep.disable(sweep);\r\n          removed += 1;\r\n        } catch (error) {\r\n          console.log(error);\r\n        }\r\n      })\r\n    );\r\n    console.log('removed sweeps:', removed);\r\n  }\r\n\r\n  //\r\n  // ---------- 3D objects (SDK Bundle only) ----------\r\n  //\r\n  async init3DObjectViewer(): Promise<void> {\r\n    return new Promise(async (resolve) => {\r\n      var [sceneObject] = await this.sdk.Scene.createObjects(1);\r\n\r\n      var node = sceneObject.addNode();\r\n\r\n      // TODO change this 🤮\r\n      node.addComponent('mp.lights');\r\n      //node.addComponent('mp.lights');\r\n      /*node.addComponent('mp.lights');\r\n      node.addComponent('mp.lights');*/\r\n      node.start();\r\n\r\n      const nodeControl = sceneObject.addNode();\r\n      this.objectControl = nodeControl.addComponent('mp.transformControls');\r\n      nodeControl.start();\r\n\r\n      //this.add3DObject({},null);\r\n\r\n      resolve();\r\n    });\r\n  }\r\n\r\n  async add3DObject(\r\n    obj: IObject3D,\r\n    mode?: 'translate' | 'rotate'\r\n  ): Promise<any> {\r\n    return new Promise(async (resolve) => {\r\n      const [sceneObject] = await this.sdk.Scene.createObjects(1);\r\n\r\n      // TODO: improvment, regroup all of these in Dynamical Objects Lib\r\n      // SecurityCamera\r\n      // NestThermostat\r\n      // Video\r\n      let isAnimatedSecurityCamera = false;\r\n      let isNestThermostat = false;\r\n      let isSmarterplanPromotionalVideo = false;\r\n      let isAzimuthalCrown = false;\r\n\r\n      /**\r\n       * TODO: refacto with an enum or switch/case\r\n       */\r\n      if (obj.object === \"security_camera\") {\r\n        isAnimatedSecurityCamera = true;\r\n      }\r\n      if (obj.object === \"nest_thermostat\") {\r\n        isNestThermostat = true;\r\n      }\r\n      if (obj.object === 'video') {\r\n        isSmarterplanPromotionalVideo = true;\r\n      }\r\n      if(obj.object === 'azimuth') {\r\n        isAzimuthalCrown = true;\r\n      }\r\n\r\n      const modelNode = sceneObject.addNode();\r\n      let component = null;\r\n      const initial = {\r\n        url: `/assets/3Dobjects/objects/${obj.object}${\r\n          obj.format.indexOf('.') === -1 ? '.' + obj.format : obj.format\r\n        }`,\r\n        // TODO/ store localPosition && localRotation in BDD too (in order to have pertfect initial placement)\r\n        localRotation: {\"x\": 0, \"y\": 0, \"z\": 0},\r\n        // TODO/ store localPosition && localRotation in BDD too (in order to have pertfect initial placement)\r\n        localPosition: {\"x\": 0, \"y\": 0, \"z\": 0},\r\n        visible: true,\r\n        colliderEnabled: true\r\n      };\r\n\r\n\r\n      switch (obj.format) {\r\n        case '.obj':\r\n        case 'obj':\r\n          component = modelNode.addComponent('mp.objLoader', initial);\r\n          break;\r\n        case '.fbx':\r\n        case 'fbx':\r\n          component = modelNode.addComponent('mp.fbxLoader', initial);\r\n          break;\r\n        case '.gltf':\r\n        case 'gltf':\r\n          component = modelNode.addComponent('mp.gltfLoader', initial);\r\n          break;\r\n        case '.glb':\r\n        case 'glb':\r\n          component = modelNode.addComponent('mp.gltfLoader', initial);\r\n          break;\r\n        case '.dae':\r\n        case 'dae':\r\n          component = modelNode.addComponent('mp.daeLoader', initial);\r\n          break;\r\n\r\n        default:\r\n          console.log('Format not supported');\r\n          break;\r\n      }\r\n\r\n      //cache system (i'll try to make it work later ...)\r\n      /*let objContentsJSON = JSON.stringify(modelNode);\r\n      console.log(modelNode);\r\n      console.log(objContentsJSON);\r\n      console.log(JSON.stringify(modelNode));\r\n      localStorage.setItem(`testObj_${obj.object}`, objContentsJSON);\r\n      console.log(\"stored ?!\");\r\n      console.log(typeof modelNode);*/\r\n      //let dataS = serialijse.serialize(modelNode);\r\n      //console.log(dataS);\r\n\r\n      // Use 3 ?! Ambient lightings\r\n      if (this.noLightForObjects) {\r\n        const lightsNode = sceneObject.addNode();\r\n        lightsNode.addComponent('mp.ambientLight', {\r\n          intensity: 1,\r\n          color: {r: 1.0, g: 1.0, b: 1.0},\r\n        });\r\n        this.noLightForObjects = false;\r\n      }\r\n\r\n      modelNode.obj3D.position.set(\r\n        obj.position.x,\r\n        obj.position.y,\r\n        obj.position.z\r\n      );\r\n\r\n      modelNode.obj3D.rotation.set(\r\n        obj.rotation.x,\r\n        obj.rotation.y,\r\n        obj.rotation.z\r\n      );\r\n      modelNode.obj3D.scale.set(obj.scale.x, obj.scale.y, obj.scale.z);\r\n      if(isAzimuthalCrown)\r\n      {\r\n        /*if(modelNode.obj3D.scale.x >= 1.0){\r\n          modelNode.obj3D.scale.set(obj.scale.x/5000, obj.scale.y/5000, obj.scale.z/5000);\r\n        }*/\r\n        this.azimuthalCrown=modelNode;\r\n        this.displayAzimutalCrown();\r\n      }\r\n\r\n      // By defaut, during creation, object has translation gizmo\r\n      // => User has to click on lateral panel, and save its position after playing with it!\r\n      if (mode && !isNestThermostat) {\r\n        this.attachGizmoControlTo3DObject(\r\n          modelNode,\r\n          sceneObject,\r\n          mode,\r\n          true,\r\n          true\r\n        ).catch((e) => console.log(e));\r\n      }\r\n\r\n      if (this.lastObject3D && typeof this.lastObject3D.id === 'string') {\r\n        // prompt ThreeJS UUID to match our last generated object if new (not from Db)\r\n        modelNode.obj3D.uuid = this.lastObject3D.id;\r\n      } else if (obj.id) {\r\n        modelNode.obj3D.uuid = obj.id;\r\n      }\r\n      this.lastObject3D = modelNode.obj3D;\r\n\r\n      // Store this in memory Map dictionnary\r\n      //console.log(\"Adding object: \");\r\n      //console.log(modelNode);\r\n      this.dictionnaryObjects3D.set(modelNode.obj3D.uuid, modelNode);\r\n      this.dictionnarySceneObjects3D.set(modelNode.obj3D.uuid, sceneObject);\r\n\r\n      /*this.sdk.Camera.pose.subscribe(\r\n        function (pose: any) {\r\n          //console.log(\"in callback\")\r\n          //console.log(this.lastCameraPosition)\r\n          //console.log(pose.position)\r\n          if((pose.position.x == this.lastCameraPosition.x && pose.position.y == this.lastCameraPosition.y && pose.position.z == this.lastCameraPosition.z) || this.inTransitionMode || this.inTransitionSweep){\r\n            //console.log(\"returning\")\r\n            return;\r\n          }\r\n          console.log(\"camera pos:\",pose.position);\r\n          this.lastCameraPosition = {...pose.position};\r\n          if(this.lastObject3D){\r\n            this.lastObject3D.position.set(pose.position.x+.5,pose.position.y+.5,pose.position.z+.5);\r\n          }\r\n        }.bind(this));*/\r\n\r\n      if (isAnimatedSecurityCamera) {\r\n\r\n        const animator = modelNode.addComponent('mp.securityCamera',\r\n          {\r\n            \"nearPlane\": 0.1,\r\n            \"farPlane\": 10,\r\n            \"horizontalFOV\": 52,\r\n            \"aspect\": 1.7777777777777777,\r\n            \"localPosition\": {\r\n              \"x\": 0.3,\r\n              \"y\": 0.18,\r\n              \"z\": 0\r\n            },\r\n            \"localRotation\": {\r\n              \"x\": -15,\r\n              \"y\": -90,\r\n              \"z\": 0\r\n            },\r\n            \"color\": 65280,\r\n            \"panPeriod\": 5,\r\n            \"panAngle\": -45\r\n          });\r\n        const modelOutput = sceneObject.addPath({\r\n          id: 'animated-model',\r\n          type: this.sdk.Scene.PathType.OUTPUT,\r\n          node: modelNode,\r\n          component: animator,\r\n          property: 'objectRoot'\r\n        });\r\n\r\n        this.securityCameraAnimator = animator;\r\n        if (!obj.viewFrustum) {\r\n          setTimeout(() => {\r\n            animator.toggleViewFrustum();\r\n          }, 1000);\r\n        }\r\n      }\r\n\r\n      if (isNestThermostat) {\r\n        // TODO: use bindPath instead using MP sdk classes (see Security Camera example)\r\n        // for TV uses CanvasImage below!\r\n        // const ci = new CanvasImage();\r\n        // ci.onInit();\r\n        const cv = new CanvasRenderer();\r\n        cv.onInit();\r\n        const plane = new PlaneRenderer();\r\n        plane.outputs = {\r\n          objectRoot: new Object3D(),\r\n          collider: new Object3D()\r\n        };\r\n        const inputTexture = cv.outputs.texture;\r\n        plane.setRootScene(this.threeJSScene);\r\n        plane.onInit(modelNode, inputTexture);\r\n        const sc = new NestThermostat();\r\n        sc.setComponent(component, plane, cv);\r\n        sc.setRootScene(this.threeJSScene);\r\n        sc.onInit(modelNode, plane, inputTexture);\r\n        cv.setCanvasNestThermostatPainter(sc);\r\n        setTimeout(() => {\r\n          sc.inputs.loadingState = \"Loaded\";\r\n          sc.onInputsUpdated();\r\n        }, 5000);\r\n      }\r\n\r\n      if (isSmarterplanPromotionalVideo) {\r\n        // TODO: use bindPath instead using MP sdk classes (see Security Camera example)\r\n        const sc = new TvPlayer();\r\n        sc.setComponent(component);\r\n        sc.onInit(modelNode);\r\n        setTimeout(() => {\r\n          sc.inputs.loadingState = \"Loaded\";\r\n          sc.onInputsUpdated();\r\n        }, 5000);\r\n      }\r\n\r\n      sceneObject.start();\r\n      resolve(this.lastObject3D);\r\n    });\r\n  }\r\n\r\n  toggleObjectVisibility(objectId: any){\r\n    let obj = this.dictionnaryObjects3D.get(objectId);\r\n    obj.obj3D.visible = !obj.obj3D.visible;\r\n  }\r\n\r\n  isObjectVisible(objectId: any): boolean {\r\n    let obj = this.dictionnaryObjects3D.get(objectId);\r\n    return obj.obj3D.visible;\r\n  }\r\n\r\n  async pointCameraTo3DObject(objectId: any){\r\n    let obj = this.dictionnaryObjects3D.get(objectId);\r\n\r\n    //We create a temporary Tag\r\n    const poiObject = {\r\n      coordinate: JSON.stringify(obj.obj3D.position),\r\n      type: PoiType.OBJECT3D,\r\n      elementID: objectId, //todo: be careful with this\r\n    } as POI;\r\n    const objectDb = {id: objectId} as DbObjectType;\r\n\r\n    try{\r\n      await this.createMattertagFromPOI(PoiType.OBJECT3D, objectDb, poiObject);\r\n    }catch(err: any){}\r\n\r\n    //Not really necessary anymore since the tag will disappear quick\r\n    //this.sdk.Tag.editOpacity(mattertagID, 0.0);//opacity);\r\n    //this.sdk.Tag.allowAction(mattertagID, {}); //disables every action\r\n\r\n    let result = this.getTagFromElementId(objectId);\r\n\r\n    await this.goToTag(result.tag);\r\n    this.deleteLastMattertag();\r\n  }\r\n\r\n  getSceneNodeFromObject3DId(uuid: string): any {\r\n    return this.dictionnarySceneObjects3D.get(uuid);\r\n  }\r\n\r\n  async displayAzimutalCrown(){\r\n    if(this.azimuthalCrown){\r\n      this.azimuthalCrown.obj3D.position.set(this.poseCamera.position.x,this.poseCamera.position.y,this.poseCamera.position.z)\r\n    }\r\n  }\r\n\r\n  async attachGizmoControlTo3DObject(\r\n    modelNode: any,\r\n    sceneObject: any,\r\n    mode: 'translate' | 'rotate' | 'scale',\r\n    visible: boolean,\r\n    isNewObject: boolean\r\n  ): Promise<any> {\r\n    // Create a scene node with a transform control component.\r\n    let node = null;\r\n    node = sceneObject.addNode();\r\n    if (!node) {\r\n      const [sceneObject] = await this.sdk.Scene.createObjects(1);\r\n      node = sceneObject.addNode();\r\n    }\r\n\r\n    const myControl = node.addComponent('mp.transformControls');\r\n    node.start();\r\n    //\r\n    // // Make the transform control visible so that the user can manipulate the control selection.\r\n    myControl.transformControls.visible = visible;\r\n    //\r\n    // // Attach the model to the transform control\r\n    myControl.inputs.selection = modelNode;\r\n    //\r\n    // // set 'translate' mode to position the selection.\r\n    myControl.inputs.mode = mode;\r\n    modelNode.obj3D.controls = myControl; // store gizmoCtrl inside object\r\n    if(isNewObject){ //i keep the current solution for new objects\r\n      if (!this.lastObject3D || !this.lastObject3D.controls) {\r\n        try {\r\n          modelNode.obj3D.uuid = this.lastObject3D.uuid || this.lastObject3D.id;\r\n        } catch (e) {\r\n          console.log(`id obj in Scene was not assigned to id from DB since`);\r\n        }\r\n        this.lastObject3D = modelNode.obj3D;\r\n      }\r\n    }else{//objects already in place have to become the \"lastObject\" (i think?)\r\n      console.log(\"in my solution !\");\r\n      console.log(modelNode);\r\n      console.log(modelNode.obj3D.uuid);\r\n      this.lastObject3D = modelNode.obj3D;\r\n    }\r\n\r\n    return modelNode;\r\n  }\r\n\r\n  removeGizmoFromLastObject(): void {\r\n    this.lastObject3D.controls.transformControls.visible = false;\r\n    this.lastObject3D.controls.transformControls.dispose();\r\n    this.lastObject3D.controls = null;\r\n  }\r\n\r\n  toggleViewFrustum(): void {\r\n    this.securityCameraAnimator.toggleViewFrustum();\r\n  }\r\n\r\n /*public set3DObjectMode(modelNode: any, mode?: string) {\r\n   if (mode) {\r\n     this.objectControl.inputs.selection = modelNode;\r\n     this.objectControl.inputs.visible = true;\r\n     this.objectControl.inputs.mode = mode;\r\n\r\n     setInterval(() => {\r\n       console.log(modelNode);\r\n     }, 5000);\r\n\r\n   } else {\r\n     this.objectControl.inputs.visible = true;\r\n   }\r\n }*/\r\n}\r\n"]}
|