@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,799 +1,799 @@
|
|
|
1
|
-
/* eslint-disable func-names */
|
|
2
|
-
/* eslint-disable class-methods-use-this */
|
|
3
|
-
import { Inject, Injectable } from '@angular/core';
|
|
4
|
-
import { getDocument, GlobalWorkerOptions } from 'pdfjs-dist';
|
|
5
|
-
import panzoom from 'panzoom';
|
|
6
|
-
import videojs from "video.js";
|
|
7
|
-
import { Subject } from 'rxjs';
|
|
8
|
-
import { CommentType, FeatureType, PoiType, } from '../../types.service';
|
|
9
|
-
import { getSignedFile, uploadFileToS3 } from '../s3.service';
|
|
10
|
-
import { getCoefficientsForImage } from '../zone-drawer.service';
|
|
11
|
-
import { getMetaForImage } from '../../helpers.service';
|
|
12
|
-
import * as i0 from "@angular/core";
|
|
13
|
-
import * as i1 from "./zone.service";
|
|
14
|
-
import * as i2 from "./navigation.service";
|
|
15
|
-
import * as i3 from "../viewer.service";
|
|
16
|
-
import * as i4 from "@angular/router";
|
|
17
|
-
import * as i5 from "../matterport.service";
|
|
18
|
-
import * as i6 from "../baseVisibility.service";
|
|
19
|
-
import * as i7 from "../content.service";
|
|
20
|
-
export class PlanService {
|
|
21
|
-
constructor(apiInjected, zoneService, navigationService,
|
|
22
|
-
// tagService: BaseTagService,
|
|
23
|
-
viewerService, router, matterportService,
|
|
24
|
-
// private config: AppConfig,
|
|
25
|
-
visibilityService, contentService) {
|
|
26
|
-
this.zoneService = zoneService;
|
|
27
|
-
this.navigationService = navigationService;
|
|
28
|
-
this.viewerService = viewerService;
|
|
29
|
-
this.router = router;
|
|
30
|
-
this.matterportService = matterportService;
|
|
31
|
-
this.visibilityService = visibilityService;
|
|
32
|
-
this.contentService = contentService;
|
|
33
|
-
this.isReady = false;
|
|
34
|
-
this.detailTagDivIsHover = false;
|
|
35
|
-
this.btnTagIsHover = false;
|
|
36
|
-
this.resizePlanSubscription = new Subject();
|
|
37
|
-
this.lastTouchTime = 0;
|
|
38
|
-
this.delayDblTouch = 500;
|
|
39
|
-
this.lastRotation = 0;
|
|
40
|
-
this.htmlContentToInject = [];
|
|
41
|
-
this.focusMouseTagDiv = false;
|
|
42
|
-
this.API = apiInjected;
|
|
43
|
-
this.matterportService.currentCameraPose.subscribe((pose) => {
|
|
44
|
-
this.updateRotation(pose.rotation.y);
|
|
45
|
-
});
|
|
46
|
-
}
|
|
47
|
-
async createPlan(planInput) {
|
|
48
|
-
return this.API.__proto__.CreatePlan(planInput);
|
|
49
|
-
}
|
|
50
|
-
async deletePlan(plan) {
|
|
51
|
-
// await deleteFromS3(plan.annexe);
|
|
52
|
-
return this.API.__proto__.DeletePlan({ id: plan.id });
|
|
53
|
-
}
|
|
54
|
-
async getPlansForSpace(spaceID) {
|
|
55
|
-
return this.API.__proto__
|
|
56
|
-
.PlansBySpace(spaceID)
|
|
57
|
-
.then((response) => response.items);
|
|
58
|
-
}
|
|
59
|
-
async getPlansForZone(zoneID) {
|
|
60
|
-
return this.API.__proto__
|
|
61
|
-
.PlansByZone(zoneID)
|
|
62
|
-
.then((response) => response.items);
|
|
63
|
-
}
|
|
64
|
-
async getSingedPlansForSpace(spaceID) {
|
|
65
|
-
const plansFromDB = await this.getPlansForSpace(spaceID);
|
|
66
|
-
// console.log(plansFromDB);
|
|
67
|
-
const plans = [];
|
|
68
|
-
if (plansFromDB.length > 0) {
|
|
69
|
-
await Promise.all(plansFromDB.map(async (plan) => {
|
|
70
|
-
const planObject = { ...plan };
|
|
71
|
-
// annexe looks like visits/modelID/plans/file.extension
|
|
72
|
-
const signed = await getSignedFile(plan.annexe);
|
|
73
|
-
if (signed) {
|
|
74
|
-
planObject.filepath = signed;
|
|
75
|
-
}
|
|
76
|
-
if (plan.annexe) {
|
|
77
|
-
const [, modelID, , filenameWithExtension] = plan.annexe.split('/');
|
|
78
|
-
const [, extention] = filenameWithExtension.split('.');
|
|
79
|
-
planObject.model3d = modelID;
|
|
80
|
-
planObject.extension = extention;
|
|
81
|
-
plans.push(planObject);
|
|
82
|
-
}
|
|
83
|
-
else {
|
|
84
|
-
console.log(`Error plan have not annexe => `);
|
|
85
|
-
console.log(plan.id);
|
|
86
|
-
}
|
|
87
|
-
}));
|
|
88
|
-
}
|
|
89
|
-
return plans;
|
|
90
|
-
}
|
|
91
|
-
async getPlansWithZonesForSpace(spaceID) {
|
|
92
|
-
const plans = await this.getPlansForSpace(spaceID);
|
|
93
|
-
const plansZones = plans.filter((plan) => plan.zone && plan.calibration);
|
|
94
|
-
return plansZones;
|
|
95
|
-
}
|
|
96
|
-
setChosenPlan(chosenPlan) {
|
|
97
|
-
this.chosenPlan = chosenPlan;
|
|
98
|
-
}
|
|
99
|
-
getChosenPlan() {
|
|
100
|
-
return this.chosenPlan;
|
|
101
|
-
}
|
|
102
|
-
setPlanFileCache(planFileCache) {
|
|
103
|
-
this.planFileCache = planFileCache;
|
|
104
|
-
return Promise.resolve();
|
|
105
|
-
}
|
|
106
|
-
getPlanFileCache() {
|
|
107
|
-
return this.planFileCache;
|
|
108
|
-
}
|
|
109
|
-
async createPlanWithAnnexe(plan, file) {
|
|
110
|
-
const createdPlan = await this.createPlan(plan);
|
|
111
|
-
const model3D = await this.zoneService.getModel3DForZone(plan.zoneID);
|
|
112
|
-
const url = await uploadFileToS3(`visits/${model3D}/plans/`, file, createdPlan.id);
|
|
113
|
-
if (url) {
|
|
114
|
-
return this.API.__proto__.UpdatePlan({ id: createdPlan.id, annexe: url });
|
|
115
|
-
}
|
|
116
|
-
return createdPlan;
|
|
117
|
-
}
|
|
118
|
-
async updatePlan(plan) {
|
|
119
|
-
return this.API.__proto__.UpdatePlan(plan);
|
|
120
|
-
}
|
|
121
|
-
async updatePlanFile() {
|
|
122
|
-
if (this.chosenPlan && this.planFileCache) {
|
|
123
|
-
const model3D = await this.zoneService.getModel3DForZone(this.chosenPlan.zoneID);
|
|
124
|
-
return uploadFileToS3(`visits/${model3D}/plans/`, this.planFileCache, this.chosenPlan.id);
|
|
125
|
-
}
|
|
126
|
-
return Promise.reject();
|
|
127
|
-
}
|
|
128
|
-
async setAllPlansForZoneNotCurrent(zoneID, currentPlanID) {
|
|
129
|
-
const plans = await this.API.__proto__.ListPlans({
|
|
130
|
-
zoneID: { eq: zoneID },
|
|
131
|
-
});
|
|
132
|
-
await Promise.all(plans.items.map(async (plan) => {
|
|
133
|
-
if (plan.isCurrentForZone && plan.id !== currentPlanID) {
|
|
134
|
-
await this.updatePlan({
|
|
135
|
-
id: plan.id,
|
|
136
|
-
isCurrentForZone: false,
|
|
137
|
-
});
|
|
138
|
-
}
|
|
139
|
-
}));
|
|
140
|
-
}
|
|
141
|
-
async configurePlan(currentPlan, divId = 'planDivPane', divContentId = 'planDivPaneContent') {
|
|
142
|
-
this.isReady = false;
|
|
143
|
-
if (!this.detailTagDiv) {
|
|
144
|
-
this.detailTagDiv = document.querySelector(`#detailTagDiv`);
|
|
145
|
-
}
|
|
146
|
-
this.detailTagDiv.addEventListener('mouseenter', () => {
|
|
147
|
-
this.focusMouseTagDiv = true;
|
|
148
|
-
});
|
|
149
|
-
this.detailTagDiv.addEventListener('mouseleave', () => {
|
|
150
|
-
this.btnTagIsHover = false;
|
|
151
|
-
this.detailTagDiv.style.display = 'none';
|
|
152
|
-
this.focusMouseTagDiv = false;
|
|
153
|
-
});
|
|
154
|
-
this.calibrationPlan = JSON.parse(currentPlan.calibration);
|
|
155
|
-
this.currentPlan = currentPlan;
|
|
156
|
-
this.planDiv = document.querySelector(`#${divId}`);
|
|
157
|
-
this.planDivContent = document.querySelector(`#${divContentId}`);
|
|
158
|
-
this.planDiv.addEventListener('touchstart', this.handleTouch.bind(this));
|
|
159
|
-
this.planDiv.addEventListener('dblclick', this.onDblClickPlan.bind(this));
|
|
160
|
-
if (!currentPlan.annexe.includes('pdf')) {
|
|
161
|
-
this.imgPlan = await getMetaForImage(currentPlan.filepath);
|
|
162
|
-
//Rect use offsetWidth : height, not actual boundingClientRect because will be modified by the zoom.
|
|
163
|
-
const rect = {
|
|
164
|
-
width: this.planDivContent.offsetWidth,
|
|
165
|
-
height: this.planDivContent.offsetHeight,
|
|
166
|
-
};
|
|
167
|
-
const { coeffX, coeffY } = getCoefficientsForImage(this.imgPlan, rect);
|
|
168
|
-
this.coeffPlanX = coeffX;
|
|
169
|
-
this.coeffPlanY = coeffY;
|
|
170
|
-
this.planDivContent.style.backgroundImage = `url(${currentPlan.filepath})`;
|
|
171
|
-
}
|
|
172
|
-
else {
|
|
173
|
-
await this.drawPdf(currentPlan);
|
|
174
|
-
}
|
|
175
|
-
this.planDivContent.style.backgroundSize = 'contain';
|
|
176
|
-
this.planDivContent.style.backgroundRepeat = 'no-repeat';
|
|
177
|
-
this.panzoom = panzoom(this.planDiv, {
|
|
178
|
-
bounds: true,
|
|
179
|
-
boundsPadding: 0,
|
|
180
|
-
maxZoom: 4,
|
|
181
|
-
zoomDoubleClickSpeed: 1, //disables double click zoom
|
|
182
|
-
//initialZoom: 4
|
|
183
|
-
});
|
|
184
|
-
this.panzoom.zoomAbs(0, 0, 4);
|
|
185
|
-
setTimeout(() => {
|
|
186
|
-
this.updateRotation(this.lastRotation);
|
|
187
|
-
}, 200);
|
|
188
|
-
return true;
|
|
189
|
-
}
|
|
190
|
-
clearPlanImage() {
|
|
191
|
-
if (this.planDivContent) {
|
|
192
|
-
this.planDivContent.style.backgroundImage = null;
|
|
193
|
-
}
|
|
194
|
-
}
|
|
195
|
-
async drawPdf(currentPlan) {
|
|
196
|
-
return new Promise(async (resolve) => {
|
|
197
|
-
const canvas = document.createElement('canvas');
|
|
198
|
-
const context = canvas.getContext('2d');
|
|
199
|
-
const pdfjsWorker = await import('pdfjs-dist/build/pdf.worker.entry');
|
|
200
|
-
GlobalWorkerOptions.workerSrc = pdfjsWorker;
|
|
201
|
-
const pdf = await getDocument(currentPlan.filepath).promise;
|
|
202
|
-
const page = await pdf.getPage(1);
|
|
203
|
-
const viewPortParameters = { scale: 1.5 };
|
|
204
|
-
const viewport = page.getViewport(viewPortParameters);
|
|
205
|
-
canvas.height = viewport.height;
|
|
206
|
-
canvas.width = viewport.width;
|
|
207
|
-
const renderContext = {
|
|
208
|
-
canvasContext: context,
|
|
209
|
-
viewport,
|
|
210
|
-
};
|
|
211
|
-
const renderTask = page.render(renderContext).promise;
|
|
212
|
-
renderTask.then(async () => {
|
|
213
|
-
const imgUrl = canvas.toDataURL('image/png');
|
|
214
|
-
this.imgPlan = await getMetaForImage(imgUrl);
|
|
215
|
-
const rect = {
|
|
216
|
-
width: this.planDivContent.offsetWidth,
|
|
217
|
-
height: this.planDivContent.offsetHeight,
|
|
218
|
-
};
|
|
219
|
-
const { coeffX, coeffY } = getCoefficientsForImage(this.imgPlan, rect);
|
|
220
|
-
this.coeffPlanX = coeffX;
|
|
221
|
-
this.coeffPlanY = coeffY;
|
|
222
|
-
this.planDiv.style.backgroundImage = `url(${imgUrl})`;
|
|
223
|
-
resolve();
|
|
224
|
-
});
|
|
225
|
-
});
|
|
226
|
-
}
|
|
227
|
-
async drawElement(element, tagType, currentPlan, tagService, config, sizeButton = 10) {
|
|
228
|
-
const [poi] = element.pois.items;
|
|
229
|
-
if (poi && poi.coordinate && currentPlan) {
|
|
230
|
-
const { zone } = currentPlan;
|
|
231
|
-
if (zone) {
|
|
232
|
-
const position = JSON.parse(poi.coordinate);
|
|
233
|
-
const button = document.createElement('button');
|
|
234
|
-
let elementTitle;
|
|
235
|
-
const url = tagService.getUrlForSeeDetails(element, tagType);
|
|
236
|
-
let tagIconImage;
|
|
237
|
-
switch (tagType) {
|
|
238
|
-
case PoiType.TICKET: {
|
|
239
|
-
tagIconImage = config.my_config.icon_ticket;
|
|
240
|
-
break;
|
|
241
|
-
}
|
|
242
|
-
case PoiType.EQUIPMENT: {
|
|
243
|
-
tagIconImage = config.my_config.icon_equipment;
|
|
244
|
-
break;
|
|
245
|
-
}
|
|
246
|
-
case PoiType.MEASURE:
|
|
247
|
-
tagIconImage = config.my_config.icon_measure;
|
|
248
|
-
break;
|
|
249
|
-
case PoiType.OBJECT3D:
|
|
250
|
-
tagIconImage = config.my_config.icon_object3d;
|
|
251
|
-
break;
|
|
252
|
-
case PoiType.DATA:
|
|
253
|
-
tagIconImage = config.my_config.icon_data;
|
|
254
|
-
break;
|
|
255
|
-
case PoiType.DESK:
|
|
256
|
-
tagIconImage = config.my_config.icon_data;
|
|
257
|
-
break;
|
|
258
|
-
default:
|
|
259
|
-
return;
|
|
260
|
-
}
|
|
261
|
-
if (poi.tagIcon) {
|
|
262
|
-
const tagIcon = JSON.parse(poi.tagIcon);
|
|
263
|
-
if (tagType === PoiType.DATA &&
|
|
264
|
-
element.type === FeatureType.INDICATOR_TEMP) {
|
|
265
|
-
tagIcon.src = tagService.getIconTagImageForFeature(element, poi);
|
|
266
|
-
}
|
|
267
|
-
// poi.tagIcon = tagIcon;
|
|
268
|
-
if (tagIcon.src) {
|
|
269
|
-
const source = await getSignedFile(tagIcon.src);
|
|
270
|
-
if (source) {
|
|
271
|
-
tagIconImage = source;
|
|
272
|
-
}
|
|
273
|
-
}
|
|
274
|
-
}
|
|
275
|
-
this.styleButton(button, tagIconImage, sizeButton);
|
|
276
|
-
button.id = element.id;
|
|
277
|
-
this.planDivContent.append(button);
|
|
278
|
-
let x;
|
|
279
|
-
let y;
|
|
280
|
-
if (this.calibrationPlan.new) {
|
|
281
|
-
// alert("new calibration plan: positioning not ready yet");
|
|
282
|
-
const position2D = this.transformPosition3DForNewCalibration(position);
|
|
283
|
-
x = position2D.x;
|
|
284
|
-
y = position2D.y;
|
|
285
|
-
button.style.top = `${y / position2D.coeffY - 5}px`;
|
|
286
|
-
button.style.left = `${x / position2D.coeffX - 5}px`;
|
|
287
|
-
}
|
|
288
|
-
else {
|
|
289
|
-
// to remove when all calibrations are new
|
|
290
|
-
x =
|
|
291
|
-
this.calibrationPlan.offsetX + position.x * this.calibrationPlan.x;
|
|
292
|
-
y =
|
|
293
|
-
this.calibrationPlan.offsetY + position.z * this.calibrationPlan.y;
|
|
294
|
-
button.style.top = `${y / this.coeffPlanY - 5}px`;
|
|
295
|
-
button.style.left = `${x / this.coeffPlanX - 5}px`;
|
|
296
|
-
}
|
|
297
|
-
const html = await tagService.getHtmlToInject(tagType, element);
|
|
298
|
-
if (html != '') {
|
|
299
|
-
this.htmlContentToInject.push({
|
|
300
|
-
elementID: element.id,
|
|
301
|
-
title: '',
|
|
302
|
-
content: html,
|
|
303
|
-
y,
|
|
304
|
-
x,
|
|
305
|
-
tagIcon: tagIconImage,
|
|
306
|
-
url,
|
|
307
|
-
});
|
|
308
|
-
}
|
|
309
|
-
else {
|
|
310
|
-
// when we don't have html => case of EMBED Comment type
|
|
311
|
-
const commentEmbed = element.comments?.items.find((com) => com.type === CommentType.EMBED && com.shownInTag);
|
|
312
|
-
if (commentEmbed) {
|
|
313
|
-
this.htmlContentToInject.push({
|
|
314
|
-
elementID: element.id,
|
|
315
|
-
title: element.title,
|
|
316
|
-
content: `<iframe src=${commentEmbed.externalLink} height="200px" width="100%"></iframe>`,
|
|
317
|
-
y,
|
|
318
|
-
x,
|
|
319
|
-
tagIcon: tagIconImage,
|
|
320
|
-
url,
|
|
321
|
-
});
|
|
322
|
-
}
|
|
323
|
-
}
|
|
324
|
-
this.addListenersToButton(button, url, tagService, element);
|
|
325
|
-
this.updateRotation(this.lastRotation);
|
|
326
|
-
}
|
|
327
|
-
}
|
|
328
|
-
}
|
|
329
|
-
async drawUserPosition(currentSweep = this.currentSweep, sizeButton = 10) {
|
|
330
|
-
this.currentSweep = currentSweep;
|
|
331
|
-
if (this.calibrationPlan && currentSweep) {
|
|
332
|
-
const position = this.matterportService.getCurrentCameraPosition()?.position;
|
|
333
|
-
if (position) {
|
|
334
|
-
if (!this.userPositionBtn) {
|
|
335
|
-
this.userPositionBtn = document.createElement('button');
|
|
336
|
-
this.userPositionBtn.id = 'currentPosition';
|
|
337
|
-
this.planDivContent.append(this.userPositionBtn);
|
|
338
|
-
this.userPositionBtn.title = 'You are here';
|
|
339
|
-
this.styleButton(this.userPositionBtn, 'https://api.iconify.design/bx:bxs-user-circle.svg?color=green&height=17&width=17', sizeButton);
|
|
340
|
-
}
|
|
341
|
-
let x;
|
|
342
|
-
let y;
|
|
343
|
-
if (this.calibrationPlan.new) {
|
|
344
|
-
const position2D = this.transformPosition3DForNewCalibration(position);
|
|
345
|
-
x = position2D.x / position2D.coeffX - 5;
|
|
346
|
-
y = position2D.y / position2D.coeffY - 5;
|
|
347
|
-
}
|
|
348
|
-
else {
|
|
349
|
-
x =
|
|
350
|
-
(this.calibrationPlan.offsetX +
|
|
351
|
-
position.x * this.calibrationPlan.x) /
|
|
352
|
-
this.coeffPlanX;
|
|
353
|
-
y =
|
|
354
|
-
(this.calibrationPlan.offsetY +
|
|
355
|
-
position.z * this.calibrationPlan.y) /
|
|
356
|
-
this.coeffPlanY;
|
|
357
|
-
}
|
|
358
|
-
this.userPositionBtn.style.top = `${y}px`;
|
|
359
|
-
this.userPositionBtn.style.left = `${x}px`;
|
|
360
|
-
this.moveOnPoint({ x: x + 5, y: y + 5 });
|
|
361
|
-
}
|
|
362
|
-
}
|
|
363
|
-
}
|
|
364
|
-
/**
|
|
365
|
-
* Center the view of the plan on coordinates
|
|
366
|
-
* @param coordinate Coordinate on the plan
|
|
367
|
-
*/
|
|
368
|
-
moveOnPoint(coordinate) {
|
|
369
|
-
if (this.planDiv) {
|
|
370
|
-
const scale = this.panzoom.getTransform().scale;
|
|
371
|
-
const moveX = this.planDiv.clientWidth / 2 - coordinate.x * scale;
|
|
372
|
-
const moveY = this.planDiv.clientHeight / 2 - coordinate.y * scale;
|
|
373
|
-
this.panzoom.moveTo(moveX, moveY);
|
|
374
|
-
this.planDivContent.style.transformOrigin = `${coordinate.x}px ${coordinate.y}px`;
|
|
375
|
-
}
|
|
376
|
-
}
|
|
377
|
-
/**
|
|
378
|
-
* Apply a rotation to the plan
|
|
379
|
-
* Buttons will remain in the correct rotation
|
|
380
|
-
* @param rotation
|
|
381
|
-
*/
|
|
382
|
-
updateRotation(rotation) {
|
|
383
|
-
this.lastRotation = rotation;
|
|
384
|
-
if (this.planDivContent) {
|
|
385
|
-
this.planDivContent.style.transform = `rotate(${rotation}deg)`;
|
|
386
|
-
const buttons = this.planDivContent.querySelectorAll('button');
|
|
387
|
-
let rotationRegex = /(rotate\(-?\d*\.?\d+deg\))/g;
|
|
388
|
-
// eslint-disable-next-line unicorn/no-array-for-each
|
|
389
|
-
buttons.forEach((button) => {
|
|
390
|
-
button.style.transform = button.style.transform.replace(rotationRegex, "") + ` rotate(${-rotation}deg)`;
|
|
391
|
-
});
|
|
392
|
-
}
|
|
393
|
-
}
|
|
394
|
-
removeCurrentPosition() {
|
|
395
|
-
const button = document.querySelector('#currentPosition');
|
|
396
|
-
if (button) {
|
|
397
|
-
button.remove();
|
|
398
|
-
this.userPositionBtn = null;
|
|
399
|
-
}
|
|
400
|
-
}
|
|
401
|
-
styleButton(button, url, sizeButton = 10) {
|
|
402
|
-
button.style.backgroundImage = `url(${url})`;
|
|
403
|
-
button.style.backgroundColor = 'transparent';
|
|
404
|
-
button.style.position = 'absolute';
|
|
405
|
-
button.style.border = 'none';
|
|
406
|
-
button.style.width = `10px`;
|
|
407
|
-
button.style.height = `10px`;
|
|
408
|
-
button.disabled = false;
|
|
409
|
-
button.style.backgroundSize = 'contain';
|
|
410
|
-
button.style.backgroundRepeat = 'no-repeat';
|
|
411
|
-
button.style.transform = `scale(${sizeButton / 10})`;
|
|
412
|
-
}
|
|
413
|
-
addListenersToButton(button, url, tagService = null, element = null) {
|
|
414
|
-
let videoJsPlayer;
|
|
415
|
-
button.addEventListener('mouseenter', async (event) => {
|
|
416
|
-
const contentForButton = this.htmlContentToInject.find((object) => object.elementID === button.id);
|
|
417
|
-
this.btnTagIsHover = true;
|
|
418
|
-
const title = document.querySelector(`#titleTagDiv`);
|
|
419
|
-
if (contentForButton.title) {
|
|
420
|
-
title.style.display = 'inline';
|
|
421
|
-
title.innerHTML = contentForButton.title;
|
|
422
|
-
}
|
|
423
|
-
else {
|
|
424
|
-
title.style.display = 'none';
|
|
425
|
-
}
|
|
426
|
-
const content = document.querySelector(`#htmlTagDiv`);
|
|
427
|
-
this.detailTagDiv.style.top = `${event.clientY + 10}px`;
|
|
428
|
-
this.detailTagDiv.style.left = `${event.clientX < 120 ? 0 : event.clientX - 120}px`;
|
|
429
|
-
this.detailTagDiv.style.display = 'block';
|
|
430
|
-
// console.log(contentForButton.content);
|
|
431
|
-
content.innerHTML = contentForButton.content;
|
|
432
|
-
const detailButton = content.querySelector('#detailBtn');
|
|
433
|
-
if (detailButton) {
|
|
434
|
-
detailButton.addEventListener('click', () => {
|
|
435
|
-
this.visibilityService.detailShowing.next(true);
|
|
436
|
-
this.router.navigate([url]);
|
|
437
|
-
});
|
|
438
|
-
}
|
|
439
|
-
// handle image carousel fullscreen button
|
|
440
|
-
const imageFullscreenBtn = content.querySelector('#image-footer');
|
|
441
|
-
if (imageFullscreenBtn) {
|
|
442
|
-
imageFullscreenBtn.addEventListener('click', function () {
|
|
443
|
-
if (tagService) {
|
|
444
|
-
tagService.onActionImageClick(contentForButton.elementID);
|
|
445
|
-
}
|
|
446
|
-
});
|
|
447
|
-
}
|
|
448
|
-
// handle booking fullscreen button
|
|
449
|
-
const bookingButton = content.querySelector('#bookingBtn');
|
|
450
|
-
if (bookingButton) {
|
|
451
|
-
bookingButton.addEventListener('click', () => {
|
|
452
|
-
this.visibilityService.detailShowing.next(true);
|
|
453
|
-
this.router.navigate([url]);
|
|
454
|
-
});
|
|
455
|
-
}
|
|
456
|
-
// handle document pdf fullscreen button
|
|
457
|
-
const docFullscreenBtn = content.querySelector('#doc-footer');
|
|
458
|
-
if (docFullscreenBtn && element) {
|
|
459
|
-
const documentUrl = tagService.getAnnexeForCommentTypeInFeature(element, CommentType.DOCUMENT);
|
|
460
|
-
docFullscreenBtn.addEventListener('click', function () {
|
|
461
|
-
if (tagService) {
|
|
462
|
-
tagService.onActionDocClick(documentUrl);
|
|
463
|
-
}
|
|
464
|
-
});
|
|
465
|
-
}
|
|
466
|
-
// handle video fullscreen button
|
|
467
|
-
const videoFullscreenBtn = content.querySelector('#btn-video-fullscreen');
|
|
468
|
-
if (videoFullscreenBtn && element) {
|
|
469
|
-
const videoUrl = tagService.getAnnexeForCommentTypeInFeature(element, CommentType.VIDEO);
|
|
470
|
-
videoFullscreenBtn.addEventListener('click', function () {
|
|
471
|
-
if (tagService) {
|
|
472
|
-
tagService.onActionVideoClick(videoUrl);
|
|
473
|
-
}
|
|
474
|
-
});
|
|
475
|
-
}
|
|
476
|
-
// handle Youtube video fullscreen button
|
|
477
|
-
const youtubeFullScreen = content.querySelector('#btn-video-fullscreen-youtube');
|
|
478
|
-
if (youtubeFullScreen && element) {
|
|
479
|
-
const youtubeUrl = tagService.getAnnexeForCommentTypeInFeature(element, CommentType.YOUTUBE);
|
|
480
|
-
youtubeFullScreen.addEventListener('click', function () {
|
|
481
|
-
if (tagService) {
|
|
482
|
-
tagService.onActionYoutubeClick(youtubeUrl);
|
|
483
|
-
}
|
|
484
|
-
});
|
|
485
|
-
}
|
|
486
|
-
// handle mute video button
|
|
487
|
-
const btnMute = document.getElementById('btn-mute');
|
|
488
|
-
const imgOn = document.getElementById('sound-on');
|
|
489
|
-
const imgOff = document.getElementById('sound-off');
|
|
490
|
-
const playerElement = document.getElementById("mus-video-tag");
|
|
491
|
-
if (playerElement) {
|
|
492
|
-
videoJsPlayer = videojs("mus-video-tag");
|
|
493
|
-
if (btnMute && videoJsPlayer) {
|
|
494
|
-
btnMute.addEventListener('click', function () {
|
|
495
|
-
if (videoJsPlayer.muted()) {
|
|
496
|
-
videoJsPlayer.muted(false);
|
|
497
|
-
if (imgOn && imgOff) {
|
|
498
|
-
imgOn.style.display = 'inline-block';
|
|
499
|
-
imgOff.style.display = 'none';
|
|
500
|
-
}
|
|
501
|
-
}
|
|
502
|
-
else {
|
|
503
|
-
videoJsPlayer.muted(true);
|
|
504
|
-
if (imgOn && imgOff) {
|
|
505
|
-
imgOn.style.display = 'none';
|
|
506
|
-
imgOff.style.display = 'inline-block';
|
|
507
|
-
}
|
|
508
|
-
}
|
|
509
|
-
});
|
|
510
|
-
}
|
|
511
|
-
}
|
|
512
|
-
// handle audio player buttons
|
|
513
|
-
const playerButton = document.querySelector('.audio-play-button');
|
|
514
|
-
const audio = document.getElementById('audio-tag');
|
|
515
|
-
const timeline = document.querySelector('.timeline-tag');
|
|
516
|
-
// const timelineValue = document.querySelector('.timeline-tag').value;
|
|
517
|
-
const soundButton = document.querySelector('.audio-sound-btn');
|
|
518
|
-
const playAudio = document.getElementById('play');
|
|
519
|
-
const pauseAudio = document.getElementById('pause');
|
|
520
|
-
const imgSoundOn = document.getElementById('audio-sound-on');
|
|
521
|
-
const imgSoundOff = document.getElementById('audio-sound-off');
|
|
522
|
-
const audioModal = document.getElementById('btn-audio-modal');
|
|
523
|
-
if (audio && playerButton && timeline) {
|
|
524
|
-
playerButton.addEventListener('click', function () {
|
|
525
|
-
if (audio.paused) {
|
|
526
|
-
audio.play();
|
|
527
|
-
pauseAudio.style.display = 'inline-block';
|
|
528
|
-
playAudio.style.display = 'none';
|
|
529
|
-
}
|
|
530
|
-
else {
|
|
531
|
-
audio.pause();
|
|
532
|
-
pauseAudio.style.display = 'none';
|
|
533
|
-
playAudio.style.display = 'inline-block';
|
|
534
|
-
}
|
|
535
|
-
});
|
|
536
|
-
soundButton.addEventListener('click', function () {
|
|
537
|
-
if (audio.muted) {
|
|
538
|
-
audio.muted = false;
|
|
539
|
-
imgSoundOn.style.display = 'inline-block';
|
|
540
|
-
imgSoundOff.style.display = 'none';
|
|
541
|
-
}
|
|
542
|
-
else {
|
|
543
|
-
audio.muted = true;
|
|
544
|
-
imgSoundOn.style.display = 'none';
|
|
545
|
-
imgSoundOff.style.display = 'inline-block';
|
|
546
|
-
}
|
|
547
|
-
});
|
|
548
|
-
function changeTimelinePosition() {
|
|
549
|
-
if (audio.duration) {
|
|
550
|
-
const percentagePosition = (100 * audio.currentTime) / audio.duration;
|
|
551
|
-
const percentagePositionString = percentagePosition + '%';
|
|
552
|
-
timeline.style.backgroundSize = percentagePositionString + ' 100%';
|
|
553
|
-
timeline.value = percentagePosition.toString();
|
|
554
|
-
}
|
|
555
|
-
}
|
|
556
|
-
audio.ontimeupdate = changeTimelinePosition;
|
|
557
|
-
function changeSeek() {
|
|
558
|
-
const time = (parseInt(timeline.value) * audio.duration) / 100;
|
|
559
|
-
audio.currentTime = time;
|
|
560
|
-
}
|
|
561
|
-
timeline.addEventListener('change', changeSeek);
|
|
562
|
-
timeline.addEventListener('input', changeSeek);
|
|
563
|
-
// handle audio click
|
|
564
|
-
if (audioModal && element) {
|
|
565
|
-
const audioComment = element.comments?.items.find((com) => com.type === CommentType.AUDIO && com.shownInTag);
|
|
566
|
-
const audioCommentID = audioComment ? audioComment.id : '';
|
|
567
|
-
audioModal.addEventListener('click', function () {
|
|
568
|
-
if (tagService) {
|
|
569
|
-
tagService.onActionAudioClick(audioCommentID);
|
|
570
|
-
}
|
|
571
|
-
});
|
|
572
|
-
}
|
|
573
|
-
}
|
|
574
|
-
});
|
|
575
|
-
button.addEventListener('mouseleave', async () => {
|
|
576
|
-
// we wait if user hovers on to div of the tag (changes this.focusMouseTagDiv)
|
|
577
|
-
this.btnTagIsHover = false;
|
|
578
|
-
setTimeout(() => {
|
|
579
|
-
if (!this.focusMouseTagDiv && !this.btnTagIsHover) {
|
|
580
|
-
this.detailTagDiv.style.display = 'none';
|
|
581
|
-
if (videoJsPlayer) {
|
|
582
|
-
videoJsPlayer.dispose();
|
|
583
|
-
}
|
|
584
|
-
}
|
|
585
|
-
}, 100);
|
|
586
|
-
});
|
|
587
|
-
button.addEventListener('click', async () => {
|
|
588
|
-
this.detailTagDiv.style.display = 'none';
|
|
589
|
-
if (videoJsPlayer) {
|
|
590
|
-
videoJsPlayer.dispose();
|
|
591
|
-
}
|
|
592
|
-
await this.viewerService.action_move_to_tag(button.id);
|
|
593
|
-
this.visibilityService.detailShowing.next(true);
|
|
594
|
-
this.router.navigate([url]);
|
|
595
|
-
});
|
|
596
|
-
}
|
|
597
|
-
clearBtn(idList) {
|
|
598
|
-
for (const id of idList) {
|
|
599
|
-
const button = document.getElementById(id);
|
|
600
|
-
if (button) {
|
|
601
|
-
button.remove();
|
|
602
|
-
}
|
|
603
|
-
}
|
|
604
|
-
}
|
|
605
|
-
clearAllButtons() {
|
|
606
|
-
if (this.planDivContent) {
|
|
607
|
-
const buttons = this.planDivContent.querySelectorAll('button');
|
|
608
|
-
// eslint-disable-next-line unicorn/no-array-for-each
|
|
609
|
-
buttons.forEach(function (currentValue) {
|
|
610
|
-
currentValue.remove();
|
|
611
|
-
});
|
|
612
|
-
this.userPositionBtn = null;
|
|
613
|
-
}
|
|
614
|
-
}
|
|
615
|
-
updateAllButtonsStyle(properties, values) {
|
|
616
|
-
if (this.planDivContent) {
|
|
617
|
-
const buttons = this.planDivContent.querySelectorAll('button');
|
|
618
|
-
// eslint-disable-next-line unicorn/no-array-for-each
|
|
619
|
-
buttons.forEach(function (currentValue) {
|
|
620
|
-
properties.forEach((property, i) => {
|
|
621
|
-
if (property === "transform") {
|
|
622
|
-
//For saving the rotation and adding scale
|
|
623
|
-
let rotationRegex = /(rotate\(-?\d*\.?\d+deg\))/g;
|
|
624
|
-
let rotationMatch = currentValue.style[property].match(rotationRegex);
|
|
625
|
-
if (rotationMatch.length > 0) {
|
|
626
|
-
currentValue.style[property] = `${values[i]} ${rotationMatch[0]}`;
|
|
627
|
-
}
|
|
628
|
-
}
|
|
629
|
-
else {
|
|
630
|
-
currentValue.style[property] = values[i];
|
|
631
|
-
}
|
|
632
|
-
});
|
|
633
|
-
});
|
|
634
|
-
}
|
|
635
|
-
}
|
|
636
|
-
onPlanRemove() {
|
|
637
|
-
this.userPositionBtn = null;
|
|
638
|
-
this.calibrationPlan = null;
|
|
639
|
-
this.currentSweep = null;
|
|
640
|
-
this.detailTagDiv = null;
|
|
641
|
-
}
|
|
642
|
-
async resizePlan(isRemoving) {
|
|
643
|
-
this.resizePlanSubscription.next(false);
|
|
644
|
-
if (isRemoving) {
|
|
645
|
-
return;
|
|
646
|
-
}
|
|
647
|
-
console.log('in resizePlan');
|
|
648
|
-
if (this.planDiv && this.currentSweep) {
|
|
649
|
-
this.panzoom = panzoom(this.planDiv, {
|
|
650
|
-
bounds: true,
|
|
651
|
-
boundsPadding: 0,
|
|
652
|
-
maxZoom: 4,
|
|
653
|
-
});
|
|
654
|
-
setTimeout(() => {
|
|
655
|
-
const rect = {
|
|
656
|
-
width: this.planDivContent.offsetWidth,
|
|
657
|
-
height: this.planDivContent.offsetHeight,
|
|
658
|
-
};
|
|
659
|
-
const { coeffX, coeffY } = getCoefficientsForImage(this.imgPlan, rect);
|
|
660
|
-
this.coeffPlanX = coeffX;
|
|
661
|
-
this.coeffPlanY = coeffY;
|
|
662
|
-
this.panzoom.zoomAbs(0, 0, 4);
|
|
663
|
-
this.resizePlanSubscription.next(true);
|
|
664
|
-
}, 200);
|
|
665
|
-
}
|
|
666
|
-
}
|
|
667
|
-
handleTouch(event) {
|
|
668
|
-
const now = Date.now();
|
|
669
|
-
if (now - this.lastTouchTime < this.delayDblTouch) {
|
|
670
|
-
this.lastTouchTime = 0;
|
|
671
|
-
// this.onDblClickPlan(event);
|
|
672
|
-
/** Simulate a dblclick for phone (we don't get offsetX/Y on touchevent) */
|
|
673
|
-
const newEvent = document.createEvent('MouseEvents');
|
|
674
|
-
const touch = event.changedTouches[0];
|
|
675
|
-
newEvent.initMouseEvent('dblclick', true, true, event.target.ownerDocument.defaultView, 0, touch.screenX, touch.screenY, touch.clientX, touch.clientY, event.ctrlKey, event.altKey, event.shirtKey, event.metaKey, 0, null);
|
|
676
|
-
event.target.dispatchEvent(newEvent);
|
|
677
|
-
}
|
|
678
|
-
else {
|
|
679
|
-
this.lastTouchTime = now;
|
|
680
|
-
}
|
|
681
|
-
}
|
|
682
|
-
async onDblClickPlan(event) {
|
|
683
|
-
event.preventDefault();
|
|
684
|
-
const clickX = event.offsetX;
|
|
685
|
-
const clickY = event.offsetY;
|
|
686
|
-
const zonePlan = this.currentPlan.zone;
|
|
687
|
-
if (zonePlan) {
|
|
688
|
-
const navigations = await this.navigationService.getNavigationsForZone(zonePlan);
|
|
689
|
-
if (navigations.length > 0) {
|
|
690
|
-
await Promise.all(navigations.map(async (nav) => {
|
|
691
|
-
const position = JSON.parse(nav.position);
|
|
692
|
-
let x;
|
|
693
|
-
let y;
|
|
694
|
-
if (this.calibrationPlan.new) {
|
|
695
|
-
const position2D = this.transformPosition3DForNewCalibration(position);
|
|
696
|
-
x = position2D.x / position2D.coeffX;
|
|
697
|
-
y = position2D.y / position2D.coeffY;
|
|
698
|
-
}
|
|
699
|
-
else {
|
|
700
|
-
x =
|
|
701
|
-
(this.calibrationPlan.offsetX +
|
|
702
|
-
position.x * this.calibrationPlan.x) /
|
|
703
|
-
this.coeffPlanX;
|
|
704
|
-
y =
|
|
705
|
-
(this.calibrationPlan.offsetY +
|
|
706
|
-
position.z * this.calibrationPlan.y) /
|
|
707
|
-
this.coeffPlanY;
|
|
708
|
-
}
|
|
709
|
-
const distX = Math.abs(clickX - x);
|
|
710
|
-
const distY = Math.abs(clickY - y);
|
|
711
|
-
nav.dist = Math.sqrt(Math.pow(distX, 2) + Math.pow(distY, 2));
|
|
712
|
-
}));
|
|
713
|
-
navigations.sort((a, b) => {
|
|
714
|
-
return a.dist - b.dist;
|
|
715
|
-
});
|
|
716
|
-
this.matterportService.action_go_to_sweep(navigations[0].matterportSweepID);
|
|
717
|
-
}
|
|
718
|
-
}
|
|
719
|
-
}
|
|
720
|
-
/**
|
|
721
|
-
* Configures plan from cache (previous plan).
|
|
722
|
-
*/
|
|
723
|
-
async uploadPlanFromCache() {
|
|
724
|
-
this.htmlContentToInject = this.cache.htmlContent;
|
|
725
|
-
await this.configurePlan(this.cache.plan);
|
|
726
|
-
return Promise.resolve();
|
|
727
|
-
}
|
|
728
|
-
/**
|
|
729
|
-
* Puts buttons (tags) from cache according to passed element IDs (for filter if any)
|
|
730
|
-
* @param elementIDs elements (tickets, equipments, etc) to be shown on plan
|
|
731
|
-
*/
|
|
732
|
-
uploadTagsFromCache(sizeButton = 10) {
|
|
733
|
-
// console.log("uploading tags from cache");
|
|
734
|
-
for (const cached of this.cache.htmlContent) {
|
|
735
|
-
const button = document.createElement('button');
|
|
736
|
-
this.styleButton(button, cached.tagIcon, sizeButton);
|
|
737
|
-
button.id = cached.elementID;
|
|
738
|
-
this.planDivContent.append(button);
|
|
739
|
-
if (this.calibrationPlan.new) {
|
|
740
|
-
const newCoeffX = this.coeffPlanX /
|
|
741
|
-
(this.imgPlan.width / this.calibrationPlan.imgWidth);
|
|
742
|
-
const newCoeffY = this.coeffPlanY /
|
|
743
|
-
(this.imgPlan.height / this.calibrationPlan.imgHeight);
|
|
744
|
-
button.style.top = `${cached.y / newCoeffY - 5}px`;
|
|
745
|
-
button.style.left = `${cached.x / newCoeffX - 5}px`;
|
|
746
|
-
}
|
|
747
|
-
else {
|
|
748
|
-
button.style.top = `${cached.y / this.coeffPlanY - 5}px`;
|
|
749
|
-
button.style.left = `${cached.x / this.coeffPlanX - 5}px`;
|
|
750
|
-
}
|
|
751
|
-
this.addListenersToButton(button, cached.url);
|
|
752
|
-
}
|
|
753
|
-
}
|
|
754
|
-
async getCalibratedPlanForZone(zone) {
|
|
755
|
-
let plans = (await this.API.PlansByZone(zone.id)).items;
|
|
756
|
-
let parentZone = null;
|
|
757
|
-
if (plans.length === 0) {
|
|
758
|
-
// try to find parent zone
|
|
759
|
-
parentZone = await this.API.GetZone(zone.parentID);
|
|
760
|
-
plans = (await this.API.PlansByZone(zone.parentID)).items;
|
|
761
|
-
}
|
|
762
|
-
const calibratedPlan = plans.find((plan) => plan.calibration && plan.isCurrentForZone);
|
|
763
|
-
if (calibratedPlan) {
|
|
764
|
-
const signed = await getSignedFile(calibratedPlan.annexe);
|
|
765
|
-
if (signed) {
|
|
766
|
-
calibratedPlan.filepath = signed;
|
|
767
|
-
}
|
|
768
|
-
calibratedPlan.navigationIDs = parentZone
|
|
769
|
-
? parentZone.sweepIDs
|
|
770
|
-
: zone.sweepIDs;
|
|
771
|
-
}
|
|
772
|
-
return calibratedPlan || null;
|
|
773
|
-
}
|
|
774
|
-
transformPosition3DForNewCalibration(position) {
|
|
775
|
-
const positionX = this.calibrationPlan.nameXAxis === 'x' ? position.x : position.z;
|
|
776
|
-
const positionY = this.calibrationPlan.nameYAxis === 'x' ? position.x : position.z;
|
|
777
|
-
let x = this.calibrationPlan.offsetX + positionX * this.calibrationPlan.scaleX;
|
|
778
|
-
let y = this.calibrationPlan.offsetY + positionY * this.calibrationPlan.scaleY;
|
|
779
|
-
const newCoeffX = this.coeffPlanX / (this.imgPlan.width / this.calibrationPlan.imgWidth);
|
|
780
|
-
// console.log("newCoeffX", newCoeffX);
|
|
781
|
-
const newCoeffY = this.coeffPlanY / (this.imgPlan.height / this.calibrationPlan.imgHeight);
|
|
782
|
-
x = (x - this.calibrationPlan.offsetXPlan) / this.calibrationPlan.scalePlan;
|
|
783
|
-
y = (y - this.calibrationPlan.offsetYPlan) / this.calibrationPlan.scalePlan;
|
|
784
|
-
// console.log(x, y);
|
|
785
|
-
return { x, y, coeffX: newCoeffX, coeffY: newCoeffY };
|
|
786
|
-
}
|
|
787
|
-
}
|
|
788
|
-
PlanService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.2.7", ngImport: i0, type: PlanService, deps: [{ token: 'currentAPIService' }, { token: i1.ZoneService }, { token: i2.NavigationService }, { token: i3.ViewerService }, { token: i4.Router }, { token: i5.MatterportService }, { token: i6.BaseVisibilityService }, { token: i7.ContentService }], target: i0.ɵɵFactoryTarget.Injectable });
|
|
789
|
-
PlanService.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "13.2.7", ngImport: i0, type: PlanService, providedIn: 'root' });
|
|
790
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.2.7", ngImport: i0, type: PlanService, decorators: [{
|
|
791
|
-
type: Injectable,
|
|
792
|
-
args: [{
|
|
793
|
-
providedIn: 'root',
|
|
794
|
-
}]
|
|
795
|
-
}], ctorParameters: function () { return [{ type: undefined, decorators: [{
|
|
796
|
-
type: Inject,
|
|
797
|
-
args: ['currentAPIService']
|
|
798
|
-
}] }, { type: i1.ZoneService }, { type: i2.NavigationService }, { type: i3.ViewerService }, { type: i4.Router }, { type: i5.MatterportService }, { type: i6.BaseVisibilityService }, { type: i7.ContentService }]; } });
|
|
799
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"plan.service.js","sourceRoot":"","sources":["../../../../../../projects/ngx-smarterplan-core/src/lib/services/models/plan.service.ts"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable func-names */\n/* eslint-disable class-methods-use-this */\nimport { Inject, Injectable } from '@angular/core';\nimport { getDocument, GlobalWorkerOptions } from 'pdfjs-dist';\nimport panzoom from 'panzoom';\nimport videojs from \"video.js\";\nimport { Subject } from 'rxjs';\nimport { Router } from '@angular/router';\nimport {\n    CommentType,\n    FeatureType,\n    Plan,\n    PoiType,\n    Zone,\n} from '../../types.service';\nimport { BaseVisibilityService } from '../baseVisibility.service';\nimport { MatterportService } from '../matterport.service';\nimport { ViewerService } from '../viewer.service';\nimport { ZoneService } from './zone.service';\nimport { NavigationService } from './navigation.service';\nimport { getSignedFile, uploadFileToS3 } from '../s3.service';\nimport { BaseTagService } from '../tag.service';\nimport { Config } from '../../config';\nimport { getCoefficientsForImage } from '../zone-drawer.service';\nimport { getMetaForImage } from '../../helpers.service';\nimport { ContentService } from '../content.service';\n\n@Injectable({\n    providedIn: 'root',\n})\nexport class PlanService {\n    chosenPlan: Plan;\n\n    currentPlan: Plan;\n\n    currentSweep: string;\n\n    planFileCache: File;\n\n    planCanvas: any;\n\n    planDiv: HTMLElement | SVGElement;\n    planDivContent: HTMLElement;\n\n    panzoom: any;\n\n    isReady: boolean = false;\n\n    userPositionBtn: HTMLButtonElement;\n\n    coeffPlanX: number;\n\n    coeffPlanY: number;\n\n    calibrationPlan: any;\n\n    detailTagDiv: HTMLElement;\n\n    detailTagDivIsHover = false;\n\n    btnTagIsHover = false;\n\n    resizePlanSubscription = new Subject<boolean>();\n\n    imgPlan: HTMLImageElement;\n\n    lastTouchTime = 0;\n\n    delayDblTouch = 500;\n\n    lastRotation: number = 0;\n\n    htmlContentToInject: {\n        title: string;\n        content: string;\n        elementID: string;\n        y: number;\n        x: number;\n        tagIcon: string;\n        url: string;\n    }[] = [];\n\n    cache: {\n        equipIDs: string[];\n        ticketIDs: string[];\n        measurementsIDs: string[];\n        featureIDs: string[];\n        zoneID: string;\n        plan: Plan;\n        htmlContent: {\n            title: string;\n            content: string;\n            elementID: string;\n            y: number;\n            x: number;\n            tagIcon: string;\n            url: string;\n        }[];\n    };\n\n    focusMouseTagDiv: boolean = false;\n\n    API: any; //AWS\n\n    constructor(\n        @Inject('currentAPIService') apiInjected: any,\n        private zoneService: ZoneService,\n        private navigationService: NavigationService,\n        // tagService: BaseTagService,\n        private viewerService: ViewerService,\n        private router: Router,\n        private matterportService: MatterportService,\n        // private config: AppConfig,\n        private visibilityService: BaseVisibilityService,\n        private contentService: ContentService\n    ) {\n        this.API = apiInjected;\n        this.matterportService.currentCameraPose.subscribe((pose) => {\n            this.updateRotation(pose.rotation.y);\n        });\n    }\n\n    async createPlan(planInput: Plan): Promise<Plan> {\n        return this.API.__proto__.CreatePlan(planInput);\n    }\n\n    async deletePlan(plan: Plan): Promise<Plan> {\n        // await deleteFromS3(plan.annexe);\n        return this.API.__proto__.DeletePlan({ id: plan.id });\n    }\n\n    async getPlansForSpace(spaceID: string): Promise<Plan[]> {\n        return this.API.__proto__\n            .PlansBySpace(spaceID)\n            .then((response) => response.items);\n    }\n\n    async getPlansForZone(zoneID: string): Promise<Plan[]> {\n        return this.API.__proto__\n            .PlansByZone(zoneID)\n            .then((response) => response.items);\n    }\n\n    async getSingedPlansForSpace(spaceID: string): Promise<Plan[]> {\n        const plansFromDB = await this.getPlansForSpace(spaceID);\n        // console.log(plansFromDB);\n        const plans = [];\n        if (plansFromDB.length > 0) {\n            await Promise.all(\n                plansFromDB.map(async (plan) => {\n                    const planObject = { ...plan } as Plan;\n                    // annexe looks like visits/modelID/plans/file.extension\n                    const signed = await getSignedFile(plan.annexe);\n                    if (signed) {\n                        planObject.filepath = signed;\n                    }\n                    if (plan.annexe) {\n                        const [, modelID, , filenameWithExtension] = plan.annexe.split('/');\n                        const [, extention] = filenameWithExtension.split('.');\n                        planObject.model3d = modelID;\n                        planObject.extension = extention;\n                        plans.push(planObject);\n                    } else {\n                        console.log(`Error plan have not annexe => `);\n                        console.log(plan.id);\n                    }\n                })\n            );\n        }\n        return plans;\n    }\n\n    async getPlansWithZonesForSpace(spaceID: string): Promise<Plan[]> {\n        const plans = await this.getPlansForSpace(spaceID);\n        const plansZones = plans.filter((plan) => plan.zone && plan.calibration);\n        return plansZones;\n    }\n\n    setChosenPlan(chosenPlan: Plan) {\n        this.chosenPlan = chosenPlan;\n    }\n\n    getChosenPlan(): Plan {\n        return this.chosenPlan;\n    }\n\n    setPlanFileCache(planFileCache: File): Promise<void> {\n        this.planFileCache = planFileCache;\n        return Promise.resolve();\n    }\n\n    getPlanFileCache(): File {\n        return this.planFileCache;\n    }\n\n    async createPlanWithAnnexe(plan: Plan, file: File): Promise<Plan> {\n        const createdPlan = await this.createPlan(plan);\n        const model3D = await this.zoneService.getModel3DForZone(plan.zoneID);\n        const url = await uploadFileToS3(\n            `visits/${model3D}/plans/`,\n            file,\n            createdPlan.id\n        );\n        if (url) {\n            return this.API.__proto__.UpdatePlan({ id: createdPlan.id, annexe: url });\n        }\n        return createdPlan;\n    }\n\n    async updatePlan(plan: any): Promise<Plan> {\n        return this.API.__proto__.UpdatePlan(plan);\n    }\n\n    async updatePlanFile(): Promise<string> {\n        if (this.chosenPlan && this.planFileCache) {\n            const model3D = await this.zoneService.getModel3DForZone(\n                this.chosenPlan.zoneID\n            );\n            return uploadFileToS3(\n                `visits/${model3D}/plans/`,\n                this.planFileCache,\n                this.chosenPlan.id\n            );\n        }\n        return Promise.reject();\n    }\n\n    async setAllPlansForZoneNotCurrent(zoneID: string, currentPlanID: string) {\n        const plans = await this.API.__proto__.ListPlans({\n            zoneID: { eq: zoneID },\n        });\n        await Promise.all(\n            plans.items.map(async (plan) => {\n                if (plan.isCurrentForZone && plan.id !== currentPlanID) {\n                    await this.updatePlan({\n                        id: plan.id,\n                        isCurrentForZone: false,\n                    });\n                }\n            })\n        );\n    }\n\n    async configurePlan(\n        currentPlan: Plan,\n        divId: string = 'planDivPane',\n        divContentId: string = 'planDivPaneContent'\n    ): Promise<boolean> {\n        this.isReady = false;\n        if (!this.detailTagDiv) {\n            this.detailTagDiv = document.querySelector(\n                `#detailTagDiv`\n            ) as HTMLElement;\n        }\n        this.detailTagDiv.addEventListener('mouseenter', () => {\n            this.focusMouseTagDiv = true;\n        });\n        this.detailTagDiv.addEventListener('mouseleave', () => {\n            this.btnTagIsHover = false;\n            this.detailTagDiv.style.display = 'none';\n            this.focusMouseTagDiv = false;\n        });\n\n        this.calibrationPlan = JSON.parse(currentPlan.calibration);\n        this.currentPlan = currentPlan;\n        this.planDiv = document.querySelector(`#${divId}`) as HTMLElement;\n        this.planDivContent = document.querySelector(\n            `#${divContentId}`\n        ) as HTMLElement;\n\n        this.planDiv.addEventListener('touchstart', this.handleTouch.bind(this));\n        this.planDiv.addEventListener('dblclick', this.onDblClickPlan.bind(this));\n\n        if (!currentPlan.annexe.includes('pdf')) {\n            this.imgPlan = await getMetaForImage(currentPlan.filepath);\n            //Rect use offsetWidth : height, not actual boundingClientRect because will be modified by the zoom.\n            const rect = {\n                width: this.planDivContent.offsetWidth,\n                height: this.planDivContent.offsetHeight,\n            };\n            const { coeffX, coeffY } = getCoefficientsForImage(this.imgPlan, rect);\n            this.coeffPlanX = coeffX;\n            this.coeffPlanY = coeffY;\n\n            this.planDivContent.style.backgroundImage = `url(${currentPlan.filepath})`;\n        } else {\n            await this.drawPdf(currentPlan);\n        }\n        this.planDivContent.style.backgroundSize = 'contain';\n        this.planDivContent.style.backgroundRepeat = 'no-repeat';\n\n        this.panzoom = panzoom(this.planDiv, {\n            bounds: true,\n            boundsPadding: 0,\n            maxZoom: 4,\n            zoomDoubleClickSpeed: 1, //disables double click zoom\n            //initialZoom: 4\n        });\n        this.panzoom.zoomAbs(0, 0, 4);\n        setTimeout(() => {\n            this.updateRotation(this.lastRotation);\n        }, 200);\n        return true;\n    }\n\n    clearPlanImage() {\n        if (this.planDivContent) {\n            this.planDivContent.style.backgroundImage = null;\n        }\n    }\n\n    async drawPdf(currentPlan: Plan): Promise<void> {\n        return new Promise(async (resolve) => {\n            const canvas = document.createElement('canvas') as HTMLCanvasElement;\n\n            const context = canvas.getContext('2d') as CanvasRenderingContext2D;\n            const pdfjsWorker = await import('pdfjs-dist/build/pdf.worker.entry');\n\n            GlobalWorkerOptions.workerSrc = pdfjsWorker;\n            const pdf = await getDocument(currentPlan.filepath).promise;\n            const page = await pdf.getPage(1);\n\n            const viewPortParameters = { scale: 1.5 };\n            const viewport = page.getViewport(viewPortParameters);\n\n            canvas.height = viewport.height;\n            canvas.width = viewport.width;\n            const renderContext = {\n                canvasContext: context,\n                viewport,\n            };\n            const renderTask = page.render(renderContext).promise;\n            renderTask.then(async () => {\n                const imgUrl = canvas.toDataURL('image/png');\n                this.imgPlan = await getMetaForImage(imgUrl);\n                const rect = {\n                    width: this.planDivContent.offsetWidth,\n                    height: this.planDivContent.offsetHeight,\n                };\n                const { coeffX, coeffY } = getCoefficientsForImage(this.imgPlan, rect);\n                this.coeffPlanX = coeffX;\n                this.coeffPlanY = coeffY;\n                this.planDiv.style.backgroundImage = `url(${imgUrl})`;\n                resolve();\n            });\n        });\n    }\n\n    async drawElement(\n        element: any,\n        tagType: PoiType,\n        currentPlan: Plan,\n        tagService: BaseTagService,\n        config: Config,\n        sizeButton: number = 10\n    ) {\n        const [poi] = element.pois.items;\n        if (poi && poi.coordinate && currentPlan) {\n            const { zone } = currentPlan;\n            if (zone) {\n                const position = JSON.parse(poi.coordinate);\n                const button = document.createElement('button');\n                let elementTitle: string;\n                const url = tagService.getUrlForSeeDetails(element, tagType);\n                let tagIconImage: string;\n                switch (tagType) {\n                    case PoiType.TICKET: {\n                        tagIconImage = config.my_config.icon_ticket;\n                        break;\n                    }\n                    case PoiType.EQUIPMENT: {\n                        tagIconImage = config.my_config.icon_equipment;\n                        break;\n                    }\n                    case PoiType.MEASURE:\n                        tagIconImage = config.my_config.icon_measure;\n                        break;\n                    case PoiType.OBJECT3D:\n                        tagIconImage = config.my_config.icon_object3d;\n                        break;\n                    case PoiType.DATA:\n                        tagIconImage = config.my_config.icon_data;\n                        break;\n                    case PoiType.DESK:\n                        tagIconImage = config.my_config.icon_data;\n                        break;\n                    default:\n                        return;\n                }\n                if (poi.tagIcon) {\n                    const tagIcon = JSON.parse(poi.tagIcon);\n                    if (\n                        tagType === PoiType.DATA &&\n                        element.type === FeatureType.INDICATOR_TEMP\n                    ) {\n                        tagIcon.src = tagService.getIconTagImageForFeature(element, poi);\n                    }\n                    // poi.tagIcon = tagIcon;\n\n                    if (tagIcon.src) {\n                        const source = await getSignedFile(tagIcon.src);\n                        if (source) {\n                            tagIconImage = source;\n                        }\n                    }\n                }\n                this.styleButton(button, tagIconImage, sizeButton);\n                button.id = element.id;\n                this.planDivContent.append(button);\n                let x: number;\n                let y: number;\n                if (this.calibrationPlan.new) {\n                    // alert(\"new calibration plan: positioning not ready yet\");\n                    const position2D =\n                        this.transformPosition3DForNewCalibration(position);\n                    x = position2D.x;\n                    y = position2D.y;\n                    button.style.top = `${y / position2D.coeffY - 5}px`;\n                    button.style.left = `${x / position2D.coeffX - 5}px`;\n                } else {\n                    // to remove when all calibrations are new\n                    x =\n                        this.calibrationPlan.offsetX + position.x * this.calibrationPlan.x;\n                    y =\n                        this.calibrationPlan.offsetY + position.z * this.calibrationPlan.y;\n                    button.style.top = `${y / this.coeffPlanY - 5}px`;\n                    button.style.left = `${x / this.coeffPlanX - 5}px`;\n                }\n\n                const html = await tagService.getHtmlToInject(tagType, element);\n                if (html != '') {\n                    this.htmlContentToInject.push({\n                        elementID: element.id,\n                        title: '',\n                        content: html,\n                        y,\n                        x,\n                        tagIcon: tagIconImage,\n                        url,\n                    });\n                } else {\n                    // when we don't have html => case of EMBED Comment type\n                    const commentEmbed = element.comments?.items.find(\n                        (com) => com.type === CommentType.EMBED && com.shownInTag\n                    );\n                    if (commentEmbed) {\n                        this.htmlContentToInject.push({\n                            elementID: element.id,\n                            title: element.title,\n                            content: `<iframe src=${commentEmbed.externalLink} height=\"200px\" width=\"100%\"></iframe>`,\n                            y,\n                            x,\n                            tagIcon: tagIconImage,\n                            url,\n                        });\n                    }\n                }\n\n                this.addListenersToButton(button, url, tagService, element);\n                this.updateRotation(this.lastRotation);\n            }\n        }\n    }\n\n    async drawUserPosition(currentSweep = this.currentSweep, sizeButton: number = 10) {\n        this.currentSweep = currentSweep;\n        if (this.calibrationPlan && currentSweep) {\n            const position =\n                this.matterportService.getCurrentCameraPosition()?.position;\n            if (position) {\n                if (!this.userPositionBtn) {\n                    this.userPositionBtn = document.createElement('button');\n                    this.userPositionBtn.id = 'currentPosition';\n                    this.planDivContent.append(this.userPositionBtn);\n                    this.userPositionBtn.title = 'You are here';\n                    this.styleButton(\n                        this.userPositionBtn,\n                        'https://api.iconify.design/bx:bxs-user-circle.svg?color=green&height=17&width=17',\n                        sizeButton\n                    );\n                }\n                let x: number;\n                let y: number;\n                if (this.calibrationPlan.new) {\n                    const position2D =\n                        this.transformPosition3DForNewCalibration(position);\n                    x = position2D.x / position2D.coeffX - 5;\n                    y = position2D.y / position2D.coeffY - 5;\n                } else {\n                    x =\n                        (this.calibrationPlan.offsetX +\n                            position.x * this.calibrationPlan.x) /\n                        this.coeffPlanX;\n                    y =\n                        (this.calibrationPlan.offsetY +\n                            position.z * this.calibrationPlan.y) /\n                        this.coeffPlanY;\n                }\n                this.userPositionBtn.style.top = `${y}px`;\n                this.userPositionBtn.style.left = `${x}px`;\n                this.moveOnPoint({ x: x + 5, y: y + 5 });\n            }\n        }\n    }\n\n    /**\n     * Center the view of the plan on coordinates\n     * @param coordinate Coordinate on the plan\n     */\n    moveOnPoint(coordinate) {\n        if (this.planDiv) {\n            const scale = this.panzoom.getTransform().scale;\n            const moveX = this.planDiv.clientWidth / 2 - coordinate.x * scale;\n            const moveY = this.planDiv.clientHeight / 2 - coordinate.y * scale;\n            this.panzoom.moveTo(moveX, moveY);\n            this.planDivContent.style.transformOrigin = `${coordinate.x}px ${coordinate.y}px`;\n        }\n    }\n\n    /**\n     * Apply a rotation to the plan\n     * Buttons will remain in the correct rotation\n     * @param rotation\n     */\n    updateRotation(rotation) {\n        this.lastRotation = rotation;\n        if (this.planDivContent) {\n            this.planDivContent.style.transform = `rotate(${rotation}deg)`;\n            const buttons = this.planDivContent.querySelectorAll('button');\n            let rotationRegex = /(rotate\\(-?\\d*\\.?\\d+deg\\))/g;\n            // eslint-disable-next-line unicorn/no-array-for-each\n            buttons.forEach((button) => {\n                button.style.transform = button.style.transform.replace(rotationRegex, \"\") + ` rotate(${-rotation}deg)`;\n            });\n        }\n    }\n\n    removeCurrentPosition() {\n        const button = document.querySelector('#currentPosition');\n        if (button) {\n            button.remove();\n            this.userPositionBtn = null;\n        }\n    }\n\n    styleButton(button: HTMLButtonElement, url: string, sizeButton: number = 10) {\n        button.style.backgroundImage = `url(${url})`;\n        button.style.backgroundColor = 'transparent';\n        button.style.position = 'absolute';\n        button.style.border = 'none';\n        button.style.width = `10px`;\n        button.style.height = `10px`;\n        button.disabled = false;\n        button.style.backgroundSize = 'contain';\n        button.style.backgroundRepeat = 'no-repeat';\n        button.style.transform = `scale(${sizeButton/10})`\n  }\n\n    addListenersToButton(\n        button: HTMLButtonElement,\n        url: string,\n        tagService: BaseTagService = null,\n        element: any = null\n    ) {\n        let videoJsPlayer;\n        button.addEventListener('mouseenter', async (event) => {\n            const contentForButton = this.htmlContentToInject.find(\n                (object) => object.elementID === button.id\n            );\n            this.btnTagIsHover = true;\n            const title = document.querySelector(`#titleTagDiv`) as HTMLElement;\n            if (contentForButton.title) {\n                title.style.display = 'inline';\n                title.innerHTML = contentForButton.title;\n            } else {\n                title.style.display = 'none';\n            }\n\n            const content = document.querySelector(`#htmlTagDiv`) as HTMLElement;\n            this.detailTagDiv.style.top = `${event.clientY + 10}px`;\n            this.detailTagDiv.style.left = `${event.clientX < 120 ? 0 : event.clientX - 120\n                }px`;\n            this.detailTagDiv.style.display = 'block';\n            // console.log(contentForButton.content);\n            content.innerHTML = contentForButton.content;\n            const detailButton = content.querySelector('#detailBtn');\n            if (detailButton) {\n                detailButton.addEventListener('click', () => {\n                    this.visibilityService.detailShowing.next(true);\n                    this.router.navigate([url]);\n                });\n            }\n\n            // handle image carousel fullscreen button\n            const imageFullscreenBtn = content.querySelector('#image-footer');\n            if (imageFullscreenBtn) {\n                imageFullscreenBtn.addEventListener('click', function () {\n                    if (tagService) {\n                        tagService.onActionImageClick(contentForButton.elementID);\n                    }\n                });\n            }\n\n            // handle booking fullscreen button\n            const bookingButton = content.querySelector('#bookingBtn');\n            if (bookingButton) {\n                bookingButton.addEventListener('click', () => {\n                    this.visibilityService.detailShowing.next(true);\n                    this.router.navigate([url]);\n                });\n            }\n\n            // handle document pdf fullscreen button\n            const docFullscreenBtn = content.querySelector('#doc-footer');\n            if (docFullscreenBtn && element) {\n                const documentUrl = tagService.getAnnexeForCommentTypeInFeature(\n                    element,\n                    CommentType.DOCUMENT\n                );\n                docFullscreenBtn.addEventListener('click', function () {\n                    if (tagService) {\n                        tagService.onActionDocClick(documentUrl);\n                    }\n                });\n            }\n\n            // handle video fullscreen button\n            const videoFullscreenBtn = content.querySelector('#btn-video-fullscreen');\n            if (videoFullscreenBtn && element) {\n                const videoUrl = tagService.getAnnexeForCommentTypeInFeature(\n                    element,\n                    CommentType.VIDEO\n                );\n                videoFullscreenBtn.addEventListener('click', function () {\n                    if (tagService) {\n                        tagService.onActionVideoClick(videoUrl);\n                    }\n                });\n            }\n\n            // handle Youtube video fullscreen button\n            const youtubeFullScreen = content.querySelector(\n                '#btn-video-fullscreen-youtube'\n            );\n            if (youtubeFullScreen && element) {\n                const youtubeUrl = tagService.getAnnexeForCommentTypeInFeature(\n                    element,\n                    CommentType.YOUTUBE\n                );\n                youtubeFullScreen.addEventListener('click', function () {\n                    if (tagService) {\n                        tagService.onActionYoutubeClick(youtubeUrl);\n                    }\n                });\n            }\n\n            // handle mute video button\n            const btnMute = document.getElementById('btn-mute');\n            const imgOn = document.getElementById('sound-on');\n            const imgOff = document.getElementById('sound-off');\n            const playerElement = document.getElementById(\"mus-video-tag\");\n\n            if (playerElement) {\n                videoJsPlayer = videojs(\"mus-video-tag\");\n                if (btnMute && videoJsPlayer) {\n                    btnMute.addEventListener('click', function () {\n                        if (videoJsPlayer.muted()) {\n                            videoJsPlayer.muted(false);\n                            if (imgOn && imgOff) {\n                                imgOn.style.display = 'inline-block';\n                                imgOff.style.display = 'none';\n                            }\n                        } else {\n                            videoJsPlayer.muted(true);\n                            if (imgOn && imgOff) {\n                                imgOn.style.display = 'none';\n                                imgOff.style.display = 'inline-block';\n                            }\n                        }\n                    });\n                }\n            }\n\n\n            // handle audio player buttons\n            const playerButton = document.querySelector('.audio-play-button');\n            const audio = document.getElementById('audio-tag') as HTMLMediaElement;\n            const timeline = document.querySelector(\n                '.timeline-tag'\n            ) as HTMLInputElement;\n            // const timelineValue = document.querySelector('.timeline-tag').value;\n            const soundButton = document.querySelector('.audio-sound-btn');\n            const playAudio = document.getElementById('play');\n            const pauseAudio = document.getElementById('pause');\n            const imgSoundOn = document.getElementById('audio-sound-on');\n            const imgSoundOff = document.getElementById('audio-sound-off');\n            const audioModal = document.getElementById('btn-audio-modal');\n            if (audio && playerButton && timeline) {\n                playerButton.addEventListener('click', function () {\n                    if (audio.paused) {\n                        audio.play();\n                        pauseAudio.style.display = 'inline-block';\n                        playAudio.style.display = 'none';\n                    } else {\n                        audio.pause();\n                        pauseAudio.style.display = 'none';\n                        playAudio.style.display = 'inline-block';\n                    }\n                });\n                soundButton.addEventListener('click', function () {\n                    if (audio.muted) {\n                        audio.muted = false;\n                        imgSoundOn.style.display = 'inline-block';\n                        imgSoundOff.style.display = 'none';\n                    } else {\n                        audio.muted = true;\n                        imgSoundOn.style.display = 'none';\n                        imgSoundOff.style.display = 'inline-block';\n                    }\n                });\n                function changeTimelinePosition() {\n                    if (audio.duration) {\n                        const percentagePosition =\n                            (100 * audio.currentTime) / audio.duration;\n                        const percentagePositionString = percentagePosition + '%';\n                        timeline.style.backgroundSize = percentagePositionString + ' 100%';\n                        timeline.value = percentagePosition.toString();\n                    }\n                }\n                audio.ontimeupdate = changeTimelinePosition;\n                function changeSeek() {\n                    const time = (parseInt(timeline.value) * audio.duration) / 100;\n                    audio.currentTime = time;\n                }\n                timeline.addEventListener('change', changeSeek);\n                timeline.addEventListener('input', changeSeek);\n                // handle audio click\n                if (audioModal && element) {\n                    const audioComment = element.comments?.items.find(\n                        (com) => com.type === CommentType.AUDIO && com.shownInTag\n                    );\n                    const audioCommentID = audioComment ? audioComment.id : '';\n                    audioModal.addEventListener('click', function () {\n                        if (tagService) {\n                            tagService.onActionAudioClick(audioCommentID);\n                        }\n                    });\n                }\n            }\n        });\n\n        button.addEventListener('mouseleave', async () => {\n            // we wait if user hovers on to div of the tag (changes this.focusMouseTagDiv)\n            this.btnTagIsHover = false;\n            setTimeout(() => {\n                if (!this.focusMouseTagDiv && !this.btnTagIsHover) {\n                    this.detailTagDiv.style.display = 'none';\n                    if (videoJsPlayer) {\n                        videoJsPlayer.dispose();\n                    }\n                }\n            }, 100);\n        });\n        button.addEventListener('click', async () => {\n            this.detailTagDiv.style.display = 'none';\n            if (videoJsPlayer) {\n                videoJsPlayer.dispose();\n            }\n            await this.viewerService.action_move_to_tag(button.id);\n            this.visibilityService.detailShowing.next(true);\n            this.router.navigate([url]);\n        });\n    }\n\n    clearBtn(idList: any[]) {\n        for (const id of idList) {\n            const button = document.getElementById(id);\n            if (button) {\n                button.remove();\n            }\n        }\n    }\n\n    clearAllButtons() {\n        if (this.planDivContent) {\n            const buttons = this.planDivContent.querySelectorAll('button');\n            // eslint-disable-next-line unicorn/no-array-for-each\n            buttons.forEach(function (currentValue) {\n                currentValue.remove();\n            });\n            this.userPositionBtn = null;\n        }\n    }\n\n    updateAllButtonsStyle(properties: Array<string>, values: Array<string>) {\n        if (this.planDivContent) {\n            const buttons = this.planDivContent.querySelectorAll('button');\n            // eslint-disable-next-line unicorn/no-array-for-each\n            buttons.forEach(function (currentValue) {\n                properties.forEach((property, i) => {\n                    if (property === \"transform\") {\n                        //For saving the rotation and adding scale\n                        let rotationRegex = /(rotate\\(-?\\d*\\.?\\d+deg\\))/g;\n                        let rotationMatch = currentValue.style[property].match(rotationRegex);\n                        if (rotationMatch.length > 0) {\n                            currentValue.style[property] = `${values[i]} ${rotationMatch[0]}`;\n                        }\n                    } else {\n                        currentValue.style[property] = values[i];\n                    }\n\n                });\n            });\n        }\n    }\n\n    onPlanRemove() {\n        this.userPositionBtn = null;\n        this.calibrationPlan = null;\n        this.currentSweep = null;\n        this.detailTagDiv = null;\n    }\n\n    async resizePlan(isRemoving: boolean) {\n        this.resizePlanSubscription.next(false);\n        if (isRemoving) {\n            return;\n        }\n        console.log('in resizePlan');\n        if (this.planDiv && this.currentSweep) {\n            this.panzoom = panzoom(this.planDiv, {\n                bounds: true,\n                boundsPadding: 0,\n                maxZoom: 4,\n            });\n            setTimeout(() => {\n                const rect = {\n                    width: this.planDivContent.offsetWidth,\n                    height: this.planDivContent.offsetHeight,\n                };\n                const { coeffX, coeffY } = getCoefficientsForImage(this.imgPlan, rect);\n                this.coeffPlanX = coeffX;\n                this.coeffPlanY = coeffY;\n                this.panzoom.zoomAbs(0, 0, 4);\n                this.resizePlanSubscription.next(true);\n            }, 200);\n        }\n    }\n\n    handleTouch(event) {\n        const now = Date.now();\n        if (now - this.lastTouchTime < this.delayDblTouch) {\n            this.lastTouchTime = 0;\n            // this.onDblClickPlan(event);\n            /** Simulate a dblclick for phone (we don't get offsetX/Y on touchevent) */\n            const newEvent = document.createEvent('MouseEvents');\n            const touch = event.changedTouches[0];\n            newEvent.initMouseEvent(\n                'dblclick',\n                true,\n                true,\n                event.target.ownerDocument.defaultView,\n                0,\n                touch.screenX,\n                touch.screenY,\n                touch.clientX,\n                touch.clientY,\n                event.ctrlKey,\n                event.altKey,\n                event.shirtKey,\n                event.metaKey,\n                0,\n                null\n            );\n            event.target.dispatchEvent(newEvent);\n        } else {\n            this.lastTouchTime = now;\n        }\n    }\n\n    async onDblClickPlan(event) {\n        event.preventDefault();\n        const clickX = event.offsetX;\n        const clickY = event.offsetY;\n        const zonePlan = this.currentPlan.zone;\n        if (zonePlan) {\n            const navigations: any =\n                await this.navigationService.getNavigationsForZone(zonePlan);\n            if (navigations.length > 0) {\n                await Promise.all(\n                    navigations.map(async (nav) => {\n                        const position = JSON.parse(nav.position);\n                        let x: number;\n                        let y: number;\n                        if (this.calibrationPlan.new) {\n                            const position2D =\n                                this.transformPosition3DForNewCalibration(position);\n                            x = position2D.x / position2D.coeffX;\n                            y = position2D.y / position2D.coeffY;\n                        } else {\n                            x =\n                                (this.calibrationPlan.offsetX +\n                                    position.x * this.calibrationPlan.x) /\n                                this.coeffPlanX;\n                            y =\n                                (this.calibrationPlan.offsetY +\n                                    position.z * this.calibrationPlan.y) /\n                                this.coeffPlanY;\n                        }\n\n                        const distX = Math.abs(clickX - x);\n                        const distY = Math.abs(clickY - y);\n                        nav.dist = Math.sqrt(Math.pow(distX, 2) + Math.pow(distY, 2));\n                    })\n                );\n                navigations.sort((a, b) => {\n                    return a.dist - b.dist;\n                });\n                this.matterportService.action_go_to_sweep(\n                    navigations[0].matterportSweepID\n                );\n            }\n        }\n    }\n\n    /**\n     * Configures plan from cache (previous plan).\n     */\n    async uploadPlanFromCache(): Promise<void> {\n        this.htmlContentToInject = this.cache.htmlContent;\n        await this.configurePlan(this.cache.plan);\n        return Promise.resolve();\n    }\n\n    /**\n     * Puts buttons (tags) from cache according to passed element IDs (for filter if any)\n     * @param elementIDs elements (tickets, equipments, etc) to be shown on plan\n     */\n    uploadTagsFromCache(sizeButton: number = 10) {\n        // console.log(\"uploading tags from cache\");\n\n        for (const cached of this.cache.htmlContent) {\n            const button = document.createElement('button');\n            this.styleButton(button, cached.tagIcon, sizeButton);\n            button.id = cached.elementID;\n            this.planDivContent.append(button);\n            if (this.calibrationPlan.new) {\n                const newCoeffX =\n                    this.coeffPlanX /\n                    (this.imgPlan.width / this.calibrationPlan.imgWidth);\n                const newCoeffY =\n                    this.coeffPlanY /\n                    (this.imgPlan.height / this.calibrationPlan.imgHeight);\n                button.style.top = `${cached.y / newCoeffY - 5}px`;\n                button.style.left = `${cached.x / newCoeffX - 5}px`;\n            } else {\n                button.style.top = `${cached.y / this.coeffPlanY - 5}px`;\n                button.style.left = `${cached.x / this.coeffPlanX - 5}px`;\n            }\n\n            this.addListenersToButton(button, cached.url);\n        }\n    }\n\n    async getCalibratedPlanForZone(zone: Zone): Promise<Plan> {\n        let plans = (await this.API.PlansByZone(zone.id)).items;\n        let parentZone = null;\n        if (plans.length === 0) {\n            // try to find parent zone\n            parentZone = await this.API.GetZone(zone.parentID);\n            plans = (await this.API.PlansByZone(zone.parentID)).items;\n        }\n        const calibratedPlan = plans.find(\n            (plan) => plan.calibration && plan.isCurrentForZone\n        ) as Plan;\n        if (calibratedPlan) {\n            const signed = await getSignedFile(calibratedPlan.annexe);\n            if (signed) {\n                calibratedPlan.filepath = signed;\n            }\n            calibratedPlan.navigationIDs = parentZone\n                ? parentZone.sweepIDs\n                : zone.sweepIDs;\n        }\n        return calibratedPlan || null;\n    }\n\n    transformPosition3DForNewCalibration(position: {\n        x: number;\n        y: number;\n        z: number;\n    }): { x: number; y: number; coeffX: number; coeffY: number; } {\n        const positionX =\n            this.calibrationPlan.nameXAxis === 'x' ? position.x : position.z;\n        const positionY =\n            this.calibrationPlan.nameYAxis === 'x' ? position.x : position.z;\n        let x =\n            this.calibrationPlan.offsetX + positionX * this.calibrationPlan.scaleX;\n        let y =\n            this.calibrationPlan.offsetY + positionY * this.calibrationPlan.scaleY;\n        const newCoeffX =\n            this.coeffPlanX / (this.imgPlan.width / this.calibrationPlan.imgWidth);\n        // console.log(\"newCoeffX\", newCoeffX);\n        const newCoeffY =\n            this.coeffPlanY / (this.imgPlan.height / this.calibrationPlan.imgHeight);\n        x = (x - this.calibrationPlan.offsetXPlan) / this.calibrationPlan.scalePlan;\n\n        y = (y - this.calibrationPlan.offsetYPlan) / this.calibrationPlan.scalePlan;\n        // console.log(x, y);\n\n        return { x, y, coeffX: newCoeffX, coeffY: newCoeffY };\n    }\n}\n"]}
|
|
1
|
+
/* eslint-disable func-names */
|
|
2
|
+
/* eslint-disable class-methods-use-this */
|
|
3
|
+
import { Inject, Injectable } from '@angular/core';
|
|
4
|
+
import { getDocument, GlobalWorkerOptions } from 'pdfjs-dist';
|
|
5
|
+
import panzoom from 'panzoom';
|
|
6
|
+
import videojs from "video.js";
|
|
7
|
+
import { Subject } from 'rxjs';
|
|
8
|
+
import { CommentType, FeatureType, PoiType, } from '../../types.service';
|
|
9
|
+
import { getSignedFile, uploadFileToS3 } from '../s3.service';
|
|
10
|
+
import { getCoefficientsForImage } from '../zone-drawer.service';
|
|
11
|
+
import { getMetaForImage } from '../../helpers.service';
|
|
12
|
+
import * as i0 from "@angular/core";
|
|
13
|
+
import * as i1 from "./zone.service";
|
|
14
|
+
import * as i2 from "./navigation.service";
|
|
15
|
+
import * as i3 from "../viewer.service";
|
|
16
|
+
import * as i4 from "@angular/router";
|
|
17
|
+
import * as i5 from "../matterport.service";
|
|
18
|
+
import * as i6 from "../baseVisibility.service";
|
|
19
|
+
import * as i7 from "../content.service";
|
|
20
|
+
export class PlanService {
|
|
21
|
+
constructor(apiInjected, zoneService, navigationService,
|
|
22
|
+
// tagService: BaseTagService,
|
|
23
|
+
viewerService, router, matterportService,
|
|
24
|
+
// private config: AppConfig,
|
|
25
|
+
visibilityService, contentService) {
|
|
26
|
+
this.zoneService = zoneService;
|
|
27
|
+
this.navigationService = navigationService;
|
|
28
|
+
this.viewerService = viewerService;
|
|
29
|
+
this.router = router;
|
|
30
|
+
this.matterportService = matterportService;
|
|
31
|
+
this.visibilityService = visibilityService;
|
|
32
|
+
this.contentService = contentService;
|
|
33
|
+
this.isReady = false;
|
|
34
|
+
this.detailTagDivIsHover = false;
|
|
35
|
+
this.btnTagIsHover = false;
|
|
36
|
+
this.resizePlanSubscription = new Subject();
|
|
37
|
+
this.lastTouchTime = 0;
|
|
38
|
+
this.delayDblTouch = 500;
|
|
39
|
+
this.lastRotation = 0;
|
|
40
|
+
this.htmlContentToInject = [];
|
|
41
|
+
this.focusMouseTagDiv = false;
|
|
42
|
+
this.API = apiInjected;
|
|
43
|
+
this.matterportService.currentCameraPose.subscribe((pose) => {
|
|
44
|
+
this.updateRotation(pose.rotation.y);
|
|
45
|
+
});
|
|
46
|
+
}
|
|
47
|
+
async createPlan(planInput) {
|
|
48
|
+
return this.API.__proto__.CreatePlan(planInput);
|
|
49
|
+
}
|
|
50
|
+
async deletePlan(plan) {
|
|
51
|
+
// await deleteFromS3(plan.annexe);
|
|
52
|
+
return this.API.__proto__.DeletePlan({ id: plan.id });
|
|
53
|
+
}
|
|
54
|
+
async getPlansForSpace(spaceID) {
|
|
55
|
+
return this.API.__proto__
|
|
56
|
+
.PlansBySpace(spaceID)
|
|
57
|
+
.then((response) => response.items);
|
|
58
|
+
}
|
|
59
|
+
async getPlansForZone(zoneID) {
|
|
60
|
+
return this.API.__proto__
|
|
61
|
+
.PlansByZone(zoneID)
|
|
62
|
+
.then((response) => response.items);
|
|
63
|
+
}
|
|
64
|
+
async getSingedPlansForSpace(spaceID) {
|
|
65
|
+
const plansFromDB = await this.getPlansForSpace(spaceID);
|
|
66
|
+
// console.log(plansFromDB);
|
|
67
|
+
const plans = [];
|
|
68
|
+
if (plansFromDB.length > 0) {
|
|
69
|
+
await Promise.all(plansFromDB.map(async (plan) => {
|
|
70
|
+
const planObject = { ...plan };
|
|
71
|
+
// annexe looks like visits/modelID/plans/file.extension
|
|
72
|
+
const signed = await getSignedFile(plan.annexe);
|
|
73
|
+
if (signed) {
|
|
74
|
+
planObject.filepath = signed;
|
|
75
|
+
}
|
|
76
|
+
if (plan.annexe) {
|
|
77
|
+
const [, modelID, , filenameWithExtension] = plan.annexe.split('/');
|
|
78
|
+
const [, extention] = filenameWithExtension.split('.');
|
|
79
|
+
planObject.model3d = modelID;
|
|
80
|
+
planObject.extension = extention;
|
|
81
|
+
plans.push(planObject);
|
|
82
|
+
}
|
|
83
|
+
else {
|
|
84
|
+
console.log(`Error plan have not annexe => `);
|
|
85
|
+
console.log(plan.id);
|
|
86
|
+
}
|
|
87
|
+
}));
|
|
88
|
+
}
|
|
89
|
+
return plans;
|
|
90
|
+
}
|
|
91
|
+
async getPlansWithZonesForSpace(spaceID) {
|
|
92
|
+
const plans = await this.getPlansForSpace(spaceID);
|
|
93
|
+
const plansZones = plans.filter((plan) => plan.zone && plan.calibration);
|
|
94
|
+
return plansZones;
|
|
95
|
+
}
|
|
96
|
+
setChosenPlan(chosenPlan) {
|
|
97
|
+
this.chosenPlan = chosenPlan;
|
|
98
|
+
}
|
|
99
|
+
getChosenPlan() {
|
|
100
|
+
return this.chosenPlan;
|
|
101
|
+
}
|
|
102
|
+
setPlanFileCache(planFileCache) {
|
|
103
|
+
this.planFileCache = planFileCache;
|
|
104
|
+
return Promise.resolve();
|
|
105
|
+
}
|
|
106
|
+
getPlanFileCache() {
|
|
107
|
+
return this.planFileCache;
|
|
108
|
+
}
|
|
109
|
+
async createPlanWithAnnexe(plan, file) {
|
|
110
|
+
const createdPlan = await this.createPlan(plan);
|
|
111
|
+
const model3D = await this.zoneService.getModel3DForZone(plan.zoneID);
|
|
112
|
+
const url = await uploadFileToS3(`visits/${model3D}/plans/`, file, createdPlan.id);
|
|
113
|
+
if (url) {
|
|
114
|
+
return this.API.__proto__.UpdatePlan({ id: createdPlan.id, annexe: url });
|
|
115
|
+
}
|
|
116
|
+
return createdPlan;
|
|
117
|
+
}
|
|
118
|
+
async updatePlan(plan) {
|
|
119
|
+
return this.API.__proto__.UpdatePlan(plan);
|
|
120
|
+
}
|
|
121
|
+
async updatePlanFile() {
|
|
122
|
+
if (this.chosenPlan && this.planFileCache) {
|
|
123
|
+
const model3D = await this.zoneService.getModel3DForZone(this.chosenPlan.zoneID);
|
|
124
|
+
return uploadFileToS3(`visits/${model3D}/plans/`, this.planFileCache, this.chosenPlan.id);
|
|
125
|
+
}
|
|
126
|
+
return Promise.reject();
|
|
127
|
+
}
|
|
128
|
+
async setAllPlansForZoneNotCurrent(zoneID, currentPlanID) {
|
|
129
|
+
const plans = await this.API.__proto__.ListPlans({
|
|
130
|
+
zoneID: { eq: zoneID },
|
|
131
|
+
});
|
|
132
|
+
await Promise.all(plans.items.map(async (plan) => {
|
|
133
|
+
if (plan.isCurrentForZone && plan.id !== currentPlanID) {
|
|
134
|
+
await this.updatePlan({
|
|
135
|
+
id: plan.id,
|
|
136
|
+
isCurrentForZone: false,
|
|
137
|
+
});
|
|
138
|
+
}
|
|
139
|
+
}));
|
|
140
|
+
}
|
|
141
|
+
async configurePlan(currentPlan, divId = 'planDivPane', divContentId = 'planDivPaneContent') {
|
|
142
|
+
this.isReady = false;
|
|
143
|
+
if (!this.detailTagDiv) {
|
|
144
|
+
this.detailTagDiv = document.querySelector(`#detailTagDiv`);
|
|
145
|
+
}
|
|
146
|
+
this.detailTagDiv.addEventListener('mouseenter', () => {
|
|
147
|
+
this.focusMouseTagDiv = true;
|
|
148
|
+
});
|
|
149
|
+
this.detailTagDiv.addEventListener('mouseleave', () => {
|
|
150
|
+
this.btnTagIsHover = false;
|
|
151
|
+
this.detailTagDiv.style.display = 'none';
|
|
152
|
+
this.focusMouseTagDiv = false;
|
|
153
|
+
});
|
|
154
|
+
this.calibrationPlan = JSON.parse(currentPlan.calibration);
|
|
155
|
+
this.currentPlan = currentPlan;
|
|
156
|
+
this.planDiv = document.querySelector(`#${divId}`);
|
|
157
|
+
this.planDivContent = document.querySelector(`#${divContentId}`);
|
|
158
|
+
this.planDiv.addEventListener('touchstart', this.handleTouch.bind(this));
|
|
159
|
+
this.planDiv.addEventListener('dblclick', this.onDblClickPlan.bind(this));
|
|
160
|
+
if (!currentPlan.annexe.includes('pdf')) {
|
|
161
|
+
this.imgPlan = await getMetaForImage(currentPlan.filepath);
|
|
162
|
+
//Rect use offsetWidth : height, not actual boundingClientRect because will be modified by the zoom.
|
|
163
|
+
const rect = {
|
|
164
|
+
width: this.planDivContent.offsetWidth,
|
|
165
|
+
height: this.planDivContent.offsetHeight,
|
|
166
|
+
};
|
|
167
|
+
const { coeffX, coeffY } = getCoefficientsForImage(this.imgPlan, rect);
|
|
168
|
+
this.coeffPlanX = coeffX;
|
|
169
|
+
this.coeffPlanY = coeffY;
|
|
170
|
+
this.planDivContent.style.backgroundImage = `url(${currentPlan.filepath})`;
|
|
171
|
+
}
|
|
172
|
+
else {
|
|
173
|
+
await this.drawPdf(currentPlan);
|
|
174
|
+
}
|
|
175
|
+
this.planDivContent.style.backgroundSize = 'contain';
|
|
176
|
+
this.planDivContent.style.backgroundRepeat = 'no-repeat';
|
|
177
|
+
this.panzoom = panzoom(this.planDiv, {
|
|
178
|
+
bounds: true,
|
|
179
|
+
boundsPadding: 0,
|
|
180
|
+
maxZoom: 4,
|
|
181
|
+
zoomDoubleClickSpeed: 1, //disables double click zoom
|
|
182
|
+
//initialZoom: 4
|
|
183
|
+
});
|
|
184
|
+
this.panzoom.zoomAbs(0, 0, 4);
|
|
185
|
+
setTimeout(() => {
|
|
186
|
+
this.updateRotation(this.lastRotation);
|
|
187
|
+
}, 200);
|
|
188
|
+
return true;
|
|
189
|
+
}
|
|
190
|
+
clearPlanImage() {
|
|
191
|
+
if (this.planDivContent) {
|
|
192
|
+
this.planDivContent.style.backgroundImage = null;
|
|
193
|
+
}
|
|
194
|
+
}
|
|
195
|
+
async drawPdf(currentPlan) {
|
|
196
|
+
return new Promise(async (resolve) => {
|
|
197
|
+
const canvas = document.createElement('canvas');
|
|
198
|
+
const context = canvas.getContext('2d');
|
|
199
|
+
const pdfjsWorker = await import('pdfjs-dist/build/pdf.worker.entry');
|
|
200
|
+
GlobalWorkerOptions.workerSrc = pdfjsWorker;
|
|
201
|
+
const pdf = await getDocument(currentPlan.filepath).promise;
|
|
202
|
+
const page = await pdf.getPage(1);
|
|
203
|
+
const viewPortParameters = { scale: 1.5 };
|
|
204
|
+
const viewport = page.getViewport(viewPortParameters);
|
|
205
|
+
canvas.height = viewport.height;
|
|
206
|
+
canvas.width = viewport.width;
|
|
207
|
+
const renderContext = {
|
|
208
|
+
canvasContext: context,
|
|
209
|
+
viewport,
|
|
210
|
+
};
|
|
211
|
+
const renderTask = page.render(renderContext).promise;
|
|
212
|
+
renderTask.then(async () => {
|
|
213
|
+
const imgUrl = canvas.toDataURL('image/png');
|
|
214
|
+
this.imgPlan = await getMetaForImage(imgUrl);
|
|
215
|
+
const rect = {
|
|
216
|
+
width: this.planDivContent.offsetWidth,
|
|
217
|
+
height: this.planDivContent.offsetHeight,
|
|
218
|
+
};
|
|
219
|
+
const { coeffX, coeffY } = getCoefficientsForImage(this.imgPlan, rect);
|
|
220
|
+
this.coeffPlanX = coeffX;
|
|
221
|
+
this.coeffPlanY = coeffY;
|
|
222
|
+
this.planDiv.style.backgroundImage = `url(${imgUrl})`;
|
|
223
|
+
resolve();
|
|
224
|
+
});
|
|
225
|
+
});
|
|
226
|
+
}
|
|
227
|
+
async drawElement(element, tagType, currentPlan, tagService, config, sizeButton = 10) {
|
|
228
|
+
const [poi] = element.pois.items;
|
|
229
|
+
if (poi && poi.coordinate && currentPlan) {
|
|
230
|
+
const { zone } = currentPlan;
|
|
231
|
+
if (zone) {
|
|
232
|
+
const position = JSON.parse(poi.coordinate);
|
|
233
|
+
const button = document.createElement('button');
|
|
234
|
+
let elementTitle;
|
|
235
|
+
const url = tagService.getUrlForSeeDetails(element, tagType);
|
|
236
|
+
let tagIconImage;
|
|
237
|
+
switch (tagType) {
|
|
238
|
+
case PoiType.TICKET: {
|
|
239
|
+
tagIconImage = config.my_config.icon_ticket;
|
|
240
|
+
break;
|
|
241
|
+
}
|
|
242
|
+
case PoiType.EQUIPMENT: {
|
|
243
|
+
tagIconImage = config.my_config.icon_equipment;
|
|
244
|
+
break;
|
|
245
|
+
}
|
|
246
|
+
case PoiType.MEASURE:
|
|
247
|
+
tagIconImage = config.my_config.icon_measure;
|
|
248
|
+
break;
|
|
249
|
+
case PoiType.OBJECT3D:
|
|
250
|
+
tagIconImage = config.my_config.icon_object3d;
|
|
251
|
+
break;
|
|
252
|
+
case PoiType.DATA:
|
|
253
|
+
tagIconImage = config.my_config.icon_data;
|
|
254
|
+
break;
|
|
255
|
+
case PoiType.DESK:
|
|
256
|
+
tagIconImage = config.my_config.icon_data;
|
|
257
|
+
break;
|
|
258
|
+
default:
|
|
259
|
+
return;
|
|
260
|
+
}
|
|
261
|
+
if (poi.tagIcon) {
|
|
262
|
+
const tagIcon = JSON.parse(poi.tagIcon);
|
|
263
|
+
if (tagType === PoiType.DATA &&
|
|
264
|
+
element.type === FeatureType.INDICATOR_TEMP) {
|
|
265
|
+
tagIcon.src = tagService.getIconTagImageForFeature(element, poi);
|
|
266
|
+
}
|
|
267
|
+
// poi.tagIcon = tagIcon;
|
|
268
|
+
if (tagIcon.src) {
|
|
269
|
+
const source = await getSignedFile(tagIcon.src);
|
|
270
|
+
if (source) {
|
|
271
|
+
tagIconImage = source;
|
|
272
|
+
}
|
|
273
|
+
}
|
|
274
|
+
}
|
|
275
|
+
this.styleButton(button, tagIconImage, sizeButton);
|
|
276
|
+
button.id = element.id;
|
|
277
|
+
this.planDivContent.append(button);
|
|
278
|
+
let x;
|
|
279
|
+
let y;
|
|
280
|
+
if (this.calibrationPlan.new) {
|
|
281
|
+
// alert("new calibration plan: positioning not ready yet");
|
|
282
|
+
const position2D = this.transformPosition3DForNewCalibration(position);
|
|
283
|
+
x = position2D.x;
|
|
284
|
+
y = position2D.y;
|
|
285
|
+
button.style.top = `${y / position2D.coeffY - 5}px`;
|
|
286
|
+
button.style.left = `${x / position2D.coeffX - 5}px`;
|
|
287
|
+
}
|
|
288
|
+
else {
|
|
289
|
+
// to remove when all calibrations are new
|
|
290
|
+
x =
|
|
291
|
+
this.calibrationPlan.offsetX + position.x * this.calibrationPlan.x;
|
|
292
|
+
y =
|
|
293
|
+
this.calibrationPlan.offsetY + position.z * this.calibrationPlan.y;
|
|
294
|
+
button.style.top = `${y / this.coeffPlanY - 5}px`;
|
|
295
|
+
button.style.left = `${x / this.coeffPlanX - 5}px`;
|
|
296
|
+
}
|
|
297
|
+
const html = await tagService.getHtmlToInject(tagType, element);
|
|
298
|
+
if (html != '') {
|
|
299
|
+
this.htmlContentToInject.push({
|
|
300
|
+
elementID: element.id,
|
|
301
|
+
title: '',
|
|
302
|
+
content: html,
|
|
303
|
+
y,
|
|
304
|
+
x,
|
|
305
|
+
tagIcon: tagIconImage,
|
|
306
|
+
url,
|
|
307
|
+
});
|
|
308
|
+
}
|
|
309
|
+
else {
|
|
310
|
+
// when we don't have html => case of EMBED Comment type
|
|
311
|
+
const commentEmbed = element.comments?.items.find((com) => com.type === CommentType.EMBED && com.shownInTag);
|
|
312
|
+
if (commentEmbed) {
|
|
313
|
+
this.htmlContentToInject.push({
|
|
314
|
+
elementID: element.id,
|
|
315
|
+
title: element.title,
|
|
316
|
+
content: `<iframe src=${commentEmbed.externalLink} height="200px" width="100%"></iframe>`,
|
|
317
|
+
y,
|
|
318
|
+
x,
|
|
319
|
+
tagIcon: tagIconImage,
|
|
320
|
+
url,
|
|
321
|
+
});
|
|
322
|
+
}
|
|
323
|
+
}
|
|
324
|
+
this.addListenersToButton(button, url, tagService, element);
|
|
325
|
+
this.updateRotation(this.lastRotation);
|
|
326
|
+
}
|
|
327
|
+
}
|
|
328
|
+
}
|
|
329
|
+
async drawUserPosition(currentSweep = this.currentSweep, sizeButton = 10) {
|
|
330
|
+
this.currentSweep = currentSweep;
|
|
331
|
+
if (this.calibrationPlan && currentSweep) {
|
|
332
|
+
const position = this.matterportService.getCurrentCameraPosition()?.position;
|
|
333
|
+
if (position) {
|
|
334
|
+
if (!this.userPositionBtn) {
|
|
335
|
+
this.userPositionBtn = document.createElement('button');
|
|
336
|
+
this.userPositionBtn.id = 'currentPosition';
|
|
337
|
+
this.planDivContent.append(this.userPositionBtn);
|
|
338
|
+
this.userPositionBtn.title = 'You are here';
|
|
339
|
+
this.styleButton(this.userPositionBtn, 'https://api.iconify.design/bx:bxs-user-circle.svg?color=green&height=17&width=17', sizeButton);
|
|
340
|
+
}
|
|
341
|
+
let x;
|
|
342
|
+
let y;
|
|
343
|
+
if (this.calibrationPlan.new) {
|
|
344
|
+
const position2D = this.transformPosition3DForNewCalibration(position);
|
|
345
|
+
x = position2D.x / position2D.coeffX - 5;
|
|
346
|
+
y = position2D.y / position2D.coeffY - 5;
|
|
347
|
+
}
|
|
348
|
+
else {
|
|
349
|
+
x =
|
|
350
|
+
(this.calibrationPlan.offsetX +
|
|
351
|
+
position.x * this.calibrationPlan.x) /
|
|
352
|
+
this.coeffPlanX;
|
|
353
|
+
y =
|
|
354
|
+
(this.calibrationPlan.offsetY +
|
|
355
|
+
position.z * this.calibrationPlan.y) /
|
|
356
|
+
this.coeffPlanY;
|
|
357
|
+
}
|
|
358
|
+
this.userPositionBtn.style.top = `${y}px`;
|
|
359
|
+
this.userPositionBtn.style.left = `${x}px`;
|
|
360
|
+
this.moveOnPoint({ x: x + 5, y: y + 5 });
|
|
361
|
+
}
|
|
362
|
+
}
|
|
363
|
+
}
|
|
364
|
+
/**
|
|
365
|
+
* Center the view of the plan on coordinates
|
|
366
|
+
* @param coordinate Coordinate on the plan
|
|
367
|
+
*/
|
|
368
|
+
moveOnPoint(coordinate) {
|
|
369
|
+
if (this.planDiv) {
|
|
370
|
+
const scale = this.panzoom.getTransform().scale;
|
|
371
|
+
const moveX = this.planDiv.clientWidth / 2 - coordinate.x * scale;
|
|
372
|
+
const moveY = this.planDiv.clientHeight / 2 - coordinate.y * scale;
|
|
373
|
+
this.panzoom.moveTo(moveX, moveY);
|
|
374
|
+
this.planDivContent.style.transformOrigin = `${coordinate.x}px ${coordinate.y}px`;
|
|
375
|
+
}
|
|
376
|
+
}
|
|
377
|
+
/**
|
|
378
|
+
* Apply a rotation to the plan
|
|
379
|
+
* Buttons will remain in the correct rotation
|
|
380
|
+
* @param rotation
|
|
381
|
+
*/
|
|
382
|
+
updateRotation(rotation) {
|
|
383
|
+
this.lastRotation = rotation;
|
|
384
|
+
if (this.planDivContent) {
|
|
385
|
+
this.planDivContent.style.transform = `rotate(${rotation}deg)`;
|
|
386
|
+
const buttons = this.planDivContent.querySelectorAll('button');
|
|
387
|
+
let rotationRegex = /(rotate\(-?\d*\.?\d+deg\))/g;
|
|
388
|
+
// eslint-disable-next-line unicorn/no-array-for-each
|
|
389
|
+
buttons.forEach((button) => {
|
|
390
|
+
button.style.transform = button.style.transform.replace(rotationRegex, "") + ` rotate(${-rotation}deg)`;
|
|
391
|
+
});
|
|
392
|
+
}
|
|
393
|
+
}
|
|
394
|
+
removeCurrentPosition() {
|
|
395
|
+
const button = document.querySelector('#currentPosition');
|
|
396
|
+
if (button) {
|
|
397
|
+
button.remove();
|
|
398
|
+
this.userPositionBtn = null;
|
|
399
|
+
}
|
|
400
|
+
}
|
|
401
|
+
styleButton(button, url, sizeButton = 10) {
|
|
402
|
+
button.style.backgroundImage = `url(${url})`;
|
|
403
|
+
button.style.backgroundColor = 'transparent';
|
|
404
|
+
button.style.position = 'absolute';
|
|
405
|
+
button.style.border = 'none';
|
|
406
|
+
button.style.width = `10px`;
|
|
407
|
+
button.style.height = `10px`;
|
|
408
|
+
button.disabled = false;
|
|
409
|
+
button.style.backgroundSize = 'contain';
|
|
410
|
+
button.style.backgroundRepeat = 'no-repeat';
|
|
411
|
+
button.style.transform = `scale(${sizeButton / 10})`;
|
|
412
|
+
}
|
|
413
|
+
addListenersToButton(button, url, tagService = null, element = null) {
|
|
414
|
+
let videoJsPlayer;
|
|
415
|
+
button.addEventListener('mouseenter', async (event) => {
|
|
416
|
+
const contentForButton = this.htmlContentToInject.find((object) => object.elementID === button.id);
|
|
417
|
+
this.btnTagIsHover = true;
|
|
418
|
+
const title = document.querySelector(`#titleTagDiv`);
|
|
419
|
+
if (contentForButton.title) {
|
|
420
|
+
title.style.display = 'inline';
|
|
421
|
+
title.innerHTML = contentForButton.title;
|
|
422
|
+
}
|
|
423
|
+
else {
|
|
424
|
+
title.style.display = 'none';
|
|
425
|
+
}
|
|
426
|
+
const content = document.querySelector(`#htmlTagDiv`);
|
|
427
|
+
this.detailTagDiv.style.top = `${event.clientY + 10}px`;
|
|
428
|
+
this.detailTagDiv.style.left = `${event.clientX < 120 ? 0 : event.clientX - 120}px`;
|
|
429
|
+
this.detailTagDiv.style.display = 'block';
|
|
430
|
+
// console.log(contentForButton.content);
|
|
431
|
+
content.innerHTML = contentForButton.content;
|
|
432
|
+
const detailButton = content.querySelector('#detailBtn');
|
|
433
|
+
if (detailButton) {
|
|
434
|
+
detailButton.addEventListener('click', () => {
|
|
435
|
+
this.visibilityService.detailShowing.next(true);
|
|
436
|
+
this.router.navigate([url]);
|
|
437
|
+
});
|
|
438
|
+
}
|
|
439
|
+
// handle image carousel fullscreen button
|
|
440
|
+
const imageFullscreenBtn = content.querySelector('#image-footer');
|
|
441
|
+
if (imageFullscreenBtn) {
|
|
442
|
+
imageFullscreenBtn.addEventListener('click', function () {
|
|
443
|
+
if (tagService) {
|
|
444
|
+
tagService.onActionImageClick(contentForButton.elementID);
|
|
445
|
+
}
|
|
446
|
+
});
|
|
447
|
+
}
|
|
448
|
+
// handle booking fullscreen button
|
|
449
|
+
const bookingButton = content.querySelector('#bookingBtn');
|
|
450
|
+
if (bookingButton) {
|
|
451
|
+
bookingButton.addEventListener('click', () => {
|
|
452
|
+
this.visibilityService.detailShowing.next(true);
|
|
453
|
+
this.router.navigate([url]);
|
|
454
|
+
});
|
|
455
|
+
}
|
|
456
|
+
// handle document pdf fullscreen button
|
|
457
|
+
const docFullscreenBtn = content.querySelector('#doc-footer');
|
|
458
|
+
if (docFullscreenBtn && element) {
|
|
459
|
+
const documentUrl = tagService.getAnnexeForCommentTypeInFeature(element, CommentType.DOCUMENT);
|
|
460
|
+
docFullscreenBtn.addEventListener('click', function () {
|
|
461
|
+
if (tagService) {
|
|
462
|
+
tagService.onActionDocClick(documentUrl);
|
|
463
|
+
}
|
|
464
|
+
});
|
|
465
|
+
}
|
|
466
|
+
// handle video fullscreen button
|
|
467
|
+
const videoFullscreenBtn = content.querySelector('#btn-video-fullscreen');
|
|
468
|
+
if (videoFullscreenBtn && element) {
|
|
469
|
+
const videoUrl = tagService.getAnnexeForCommentTypeInFeature(element, CommentType.VIDEO);
|
|
470
|
+
videoFullscreenBtn.addEventListener('click', function () {
|
|
471
|
+
if (tagService) {
|
|
472
|
+
tagService.onActionVideoClick(videoUrl);
|
|
473
|
+
}
|
|
474
|
+
});
|
|
475
|
+
}
|
|
476
|
+
// handle Youtube video fullscreen button
|
|
477
|
+
const youtubeFullScreen = content.querySelector('#btn-video-fullscreen-youtube');
|
|
478
|
+
if (youtubeFullScreen && element) {
|
|
479
|
+
const youtubeUrl = tagService.getAnnexeForCommentTypeInFeature(element, CommentType.YOUTUBE);
|
|
480
|
+
youtubeFullScreen.addEventListener('click', function () {
|
|
481
|
+
if (tagService) {
|
|
482
|
+
tagService.onActionYoutubeClick(youtubeUrl);
|
|
483
|
+
}
|
|
484
|
+
});
|
|
485
|
+
}
|
|
486
|
+
// handle mute video button
|
|
487
|
+
const btnMute = document.getElementById('btn-mute');
|
|
488
|
+
const imgOn = document.getElementById('sound-on');
|
|
489
|
+
const imgOff = document.getElementById('sound-off');
|
|
490
|
+
const playerElement = document.getElementById("mus-video-tag");
|
|
491
|
+
if (playerElement) {
|
|
492
|
+
videoJsPlayer = videojs("mus-video-tag");
|
|
493
|
+
if (btnMute && videoJsPlayer) {
|
|
494
|
+
btnMute.addEventListener('click', function () {
|
|
495
|
+
if (videoJsPlayer.muted()) {
|
|
496
|
+
videoJsPlayer.muted(false);
|
|
497
|
+
if (imgOn && imgOff) {
|
|
498
|
+
imgOn.style.display = 'inline-block';
|
|
499
|
+
imgOff.style.display = 'none';
|
|
500
|
+
}
|
|
501
|
+
}
|
|
502
|
+
else {
|
|
503
|
+
videoJsPlayer.muted(true);
|
|
504
|
+
if (imgOn && imgOff) {
|
|
505
|
+
imgOn.style.display = 'none';
|
|
506
|
+
imgOff.style.display = 'inline-block';
|
|
507
|
+
}
|
|
508
|
+
}
|
|
509
|
+
});
|
|
510
|
+
}
|
|
511
|
+
}
|
|
512
|
+
// handle audio player buttons
|
|
513
|
+
const playerButton = document.querySelector('.audio-play-button');
|
|
514
|
+
const audio = document.getElementById('audio-tag');
|
|
515
|
+
const timeline = document.querySelector('.timeline-tag');
|
|
516
|
+
// const timelineValue = document.querySelector('.timeline-tag').value;
|
|
517
|
+
const soundButton = document.querySelector('.audio-sound-btn');
|
|
518
|
+
const playAudio = document.getElementById('play');
|
|
519
|
+
const pauseAudio = document.getElementById('pause');
|
|
520
|
+
const imgSoundOn = document.getElementById('audio-sound-on');
|
|
521
|
+
const imgSoundOff = document.getElementById('audio-sound-off');
|
|
522
|
+
const audioModal = document.getElementById('btn-audio-modal');
|
|
523
|
+
if (audio && playerButton && timeline) {
|
|
524
|
+
playerButton.addEventListener('click', function () {
|
|
525
|
+
if (audio.paused) {
|
|
526
|
+
audio.play();
|
|
527
|
+
pauseAudio.style.display = 'inline-block';
|
|
528
|
+
playAudio.style.display = 'none';
|
|
529
|
+
}
|
|
530
|
+
else {
|
|
531
|
+
audio.pause();
|
|
532
|
+
pauseAudio.style.display = 'none';
|
|
533
|
+
playAudio.style.display = 'inline-block';
|
|
534
|
+
}
|
|
535
|
+
});
|
|
536
|
+
soundButton.addEventListener('click', function () {
|
|
537
|
+
if (audio.muted) {
|
|
538
|
+
audio.muted = false;
|
|
539
|
+
imgSoundOn.style.display = 'inline-block';
|
|
540
|
+
imgSoundOff.style.display = 'none';
|
|
541
|
+
}
|
|
542
|
+
else {
|
|
543
|
+
audio.muted = true;
|
|
544
|
+
imgSoundOn.style.display = 'none';
|
|
545
|
+
imgSoundOff.style.display = 'inline-block';
|
|
546
|
+
}
|
|
547
|
+
});
|
|
548
|
+
function changeTimelinePosition() {
|
|
549
|
+
if (audio.duration) {
|
|
550
|
+
const percentagePosition = (100 * audio.currentTime) / audio.duration;
|
|
551
|
+
const percentagePositionString = percentagePosition + '%';
|
|
552
|
+
timeline.style.backgroundSize = percentagePositionString + ' 100%';
|
|
553
|
+
timeline.value = percentagePosition.toString();
|
|
554
|
+
}
|
|
555
|
+
}
|
|
556
|
+
audio.ontimeupdate = changeTimelinePosition;
|
|
557
|
+
function changeSeek() {
|
|
558
|
+
const time = (parseInt(timeline.value) * audio.duration) / 100;
|
|
559
|
+
audio.currentTime = time;
|
|
560
|
+
}
|
|
561
|
+
timeline.addEventListener('change', changeSeek);
|
|
562
|
+
timeline.addEventListener('input', changeSeek);
|
|
563
|
+
// handle audio click
|
|
564
|
+
if (audioModal && element) {
|
|
565
|
+
const audioComment = element.comments?.items.find((com) => com.type === CommentType.AUDIO && com.shownInTag);
|
|
566
|
+
const audioCommentID = audioComment ? audioComment.id : '';
|
|
567
|
+
audioModal.addEventListener('click', function () {
|
|
568
|
+
if (tagService) {
|
|
569
|
+
tagService.onActionAudioClick(audioCommentID);
|
|
570
|
+
}
|
|
571
|
+
});
|
|
572
|
+
}
|
|
573
|
+
}
|
|
574
|
+
});
|
|
575
|
+
button.addEventListener('mouseleave', async () => {
|
|
576
|
+
// we wait if user hovers on to div of the tag (changes this.focusMouseTagDiv)
|
|
577
|
+
this.btnTagIsHover = false;
|
|
578
|
+
setTimeout(() => {
|
|
579
|
+
if (!this.focusMouseTagDiv && !this.btnTagIsHover) {
|
|
580
|
+
this.detailTagDiv.style.display = 'none';
|
|
581
|
+
if (videoJsPlayer) {
|
|
582
|
+
videoJsPlayer.dispose();
|
|
583
|
+
}
|
|
584
|
+
}
|
|
585
|
+
}, 100);
|
|
586
|
+
});
|
|
587
|
+
button.addEventListener('click', async () => {
|
|
588
|
+
this.detailTagDiv.style.display = 'none';
|
|
589
|
+
if (videoJsPlayer) {
|
|
590
|
+
videoJsPlayer.dispose();
|
|
591
|
+
}
|
|
592
|
+
await this.viewerService.action_move_to_tag(button.id);
|
|
593
|
+
this.visibilityService.detailShowing.next(true);
|
|
594
|
+
this.router.navigate([url]);
|
|
595
|
+
});
|
|
596
|
+
}
|
|
597
|
+
clearBtn(idList) {
|
|
598
|
+
for (const id of idList) {
|
|
599
|
+
const button = document.getElementById(id);
|
|
600
|
+
if (button) {
|
|
601
|
+
button.remove();
|
|
602
|
+
}
|
|
603
|
+
}
|
|
604
|
+
}
|
|
605
|
+
clearAllButtons() {
|
|
606
|
+
if (this.planDivContent) {
|
|
607
|
+
const buttons = this.planDivContent.querySelectorAll('button');
|
|
608
|
+
// eslint-disable-next-line unicorn/no-array-for-each
|
|
609
|
+
buttons.forEach(function (currentValue) {
|
|
610
|
+
currentValue.remove();
|
|
611
|
+
});
|
|
612
|
+
this.userPositionBtn = null;
|
|
613
|
+
}
|
|
614
|
+
}
|
|
615
|
+
updateAllButtonsStyle(properties, values) {
|
|
616
|
+
if (this.planDivContent) {
|
|
617
|
+
const buttons = this.planDivContent.querySelectorAll('button');
|
|
618
|
+
// eslint-disable-next-line unicorn/no-array-for-each
|
|
619
|
+
buttons.forEach(function (currentValue) {
|
|
620
|
+
properties.forEach((property, i) => {
|
|
621
|
+
if (property === "transform") {
|
|
622
|
+
//For saving the rotation and adding scale
|
|
623
|
+
let rotationRegex = /(rotate\(-?\d*\.?\d+deg\))/g;
|
|
624
|
+
let rotationMatch = currentValue.style[property].match(rotationRegex);
|
|
625
|
+
if (rotationMatch.length > 0) {
|
|
626
|
+
currentValue.style[property] = `${values[i]} ${rotationMatch[0]}`;
|
|
627
|
+
}
|
|
628
|
+
}
|
|
629
|
+
else {
|
|
630
|
+
currentValue.style[property] = values[i];
|
|
631
|
+
}
|
|
632
|
+
});
|
|
633
|
+
});
|
|
634
|
+
}
|
|
635
|
+
}
|
|
636
|
+
onPlanRemove() {
|
|
637
|
+
this.userPositionBtn = null;
|
|
638
|
+
this.calibrationPlan = null;
|
|
639
|
+
this.currentSweep = null;
|
|
640
|
+
this.detailTagDiv = null;
|
|
641
|
+
}
|
|
642
|
+
async resizePlan(isRemoving) {
|
|
643
|
+
this.resizePlanSubscription.next(false);
|
|
644
|
+
if (isRemoving) {
|
|
645
|
+
return;
|
|
646
|
+
}
|
|
647
|
+
console.log('in resizePlan');
|
|
648
|
+
if (this.planDiv && this.currentSweep) {
|
|
649
|
+
this.panzoom = panzoom(this.planDiv, {
|
|
650
|
+
bounds: true,
|
|
651
|
+
boundsPadding: 0,
|
|
652
|
+
maxZoom: 4,
|
|
653
|
+
});
|
|
654
|
+
setTimeout(() => {
|
|
655
|
+
const rect = {
|
|
656
|
+
width: this.planDivContent.offsetWidth,
|
|
657
|
+
height: this.planDivContent.offsetHeight,
|
|
658
|
+
};
|
|
659
|
+
const { coeffX, coeffY } = getCoefficientsForImage(this.imgPlan, rect);
|
|
660
|
+
this.coeffPlanX = coeffX;
|
|
661
|
+
this.coeffPlanY = coeffY;
|
|
662
|
+
this.panzoom.zoomAbs(0, 0, 4);
|
|
663
|
+
this.resizePlanSubscription.next(true);
|
|
664
|
+
}, 200);
|
|
665
|
+
}
|
|
666
|
+
}
|
|
667
|
+
handleTouch(event) {
|
|
668
|
+
const now = Date.now();
|
|
669
|
+
if (now - this.lastTouchTime < this.delayDblTouch) {
|
|
670
|
+
this.lastTouchTime = 0;
|
|
671
|
+
// this.onDblClickPlan(event);
|
|
672
|
+
/** Simulate a dblclick for phone (we don't get offsetX/Y on touchevent) */
|
|
673
|
+
const newEvent = document.createEvent('MouseEvents');
|
|
674
|
+
const touch = event.changedTouches[0];
|
|
675
|
+
newEvent.initMouseEvent('dblclick', true, true, event.target.ownerDocument.defaultView, 0, touch.screenX, touch.screenY, touch.clientX, touch.clientY, event.ctrlKey, event.altKey, event.shirtKey, event.metaKey, 0, null);
|
|
676
|
+
event.target.dispatchEvent(newEvent);
|
|
677
|
+
}
|
|
678
|
+
else {
|
|
679
|
+
this.lastTouchTime = now;
|
|
680
|
+
}
|
|
681
|
+
}
|
|
682
|
+
async onDblClickPlan(event) {
|
|
683
|
+
event.preventDefault();
|
|
684
|
+
const clickX = event.offsetX;
|
|
685
|
+
const clickY = event.offsetY;
|
|
686
|
+
const zonePlan = this.currentPlan.zone;
|
|
687
|
+
if (zonePlan) {
|
|
688
|
+
const navigations = await this.navigationService.getNavigationsForZone(zonePlan);
|
|
689
|
+
if (navigations.length > 0) {
|
|
690
|
+
await Promise.all(navigations.map(async (nav) => {
|
|
691
|
+
const position = JSON.parse(nav.position);
|
|
692
|
+
let x;
|
|
693
|
+
let y;
|
|
694
|
+
if (this.calibrationPlan.new) {
|
|
695
|
+
const position2D = this.transformPosition3DForNewCalibration(position);
|
|
696
|
+
x = position2D.x / position2D.coeffX;
|
|
697
|
+
y = position2D.y / position2D.coeffY;
|
|
698
|
+
}
|
|
699
|
+
else {
|
|
700
|
+
x =
|
|
701
|
+
(this.calibrationPlan.offsetX +
|
|
702
|
+
position.x * this.calibrationPlan.x) /
|
|
703
|
+
this.coeffPlanX;
|
|
704
|
+
y =
|
|
705
|
+
(this.calibrationPlan.offsetY +
|
|
706
|
+
position.z * this.calibrationPlan.y) /
|
|
707
|
+
this.coeffPlanY;
|
|
708
|
+
}
|
|
709
|
+
const distX = Math.abs(clickX - x);
|
|
710
|
+
const distY = Math.abs(clickY - y);
|
|
711
|
+
nav.dist = Math.sqrt(Math.pow(distX, 2) + Math.pow(distY, 2));
|
|
712
|
+
}));
|
|
713
|
+
navigations.sort((a, b) => {
|
|
714
|
+
return a.dist - b.dist;
|
|
715
|
+
});
|
|
716
|
+
this.matterportService.action_go_to_sweep(navigations[0].matterportSweepID);
|
|
717
|
+
}
|
|
718
|
+
}
|
|
719
|
+
}
|
|
720
|
+
/**
|
|
721
|
+
* Configures plan from cache (previous plan).
|
|
722
|
+
*/
|
|
723
|
+
async uploadPlanFromCache() {
|
|
724
|
+
this.htmlContentToInject = this.cache.htmlContent;
|
|
725
|
+
await this.configurePlan(this.cache.plan);
|
|
726
|
+
return Promise.resolve();
|
|
727
|
+
}
|
|
728
|
+
/**
|
|
729
|
+
* Puts buttons (tags) from cache according to passed element IDs (for filter if any)
|
|
730
|
+
* @param elementIDs elements (tickets, equipments, etc) to be shown on plan
|
|
731
|
+
*/
|
|
732
|
+
uploadTagsFromCache(sizeButton = 10) {
|
|
733
|
+
// console.log("uploading tags from cache");
|
|
734
|
+
for (const cached of this.cache.htmlContent) {
|
|
735
|
+
const button = document.createElement('button');
|
|
736
|
+
this.styleButton(button, cached.tagIcon, sizeButton);
|
|
737
|
+
button.id = cached.elementID;
|
|
738
|
+
this.planDivContent.append(button);
|
|
739
|
+
if (this.calibrationPlan.new) {
|
|
740
|
+
const newCoeffX = this.coeffPlanX /
|
|
741
|
+
(this.imgPlan.width / this.calibrationPlan.imgWidth);
|
|
742
|
+
const newCoeffY = this.coeffPlanY /
|
|
743
|
+
(this.imgPlan.height / this.calibrationPlan.imgHeight);
|
|
744
|
+
button.style.top = `${cached.y / newCoeffY - 5}px`;
|
|
745
|
+
button.style.left = `${cached.x / newCoeffX - 5}px`;
|
|
746
|
+
}
|
|
747
|
+
else {
|
|
748
|
+
button.style.top = `${cached.y / this.coeffPlanY - 5}px`;
|
|
749
|
+
button.style.left = `${cached.x / this.coeffPlanX - 5}px`;
|
|
750
|
+
}
|
|
751
|
+
this.addListenersToButton(button, cached.url);
|
|
752
|
+
}
|
|
753
|
+
}
|
|
754
|
+
async getCalibratedPlanForZone(zone) {
|
|
755
|
+
let plans = (await this.API.PlansByZone(zone.id)).items;
|
|
756
|
+
let parentZone = null;
|
|
757
|
+
if (plans.length === 0) {
|
|
758
|
+
// try to find parent zone
|
|
759
|
+
parentZone = await this.API.GetZone(zone.parentID);
|
|
760
|
+
plans = (await this.API.PlansByZone(zone.parentID)).items;
|
|
761
|
+
}
|
|
762
|
+
const calibratedPlan = plans.find((plan) => plan.calibration && plan.isCurrentForZone);
|
|
763
|
+
if (calibratedPlan) {
|
|
764
|
+
const signed = await getSignedFile(calibratedPlan.annexe);
|
|
765
|
+
if (signed) {
|
|
766
|
+
calibratedPlan.filepath = signed;
|
|
767
|
+
}
|
|
768
|
+
calibratedPlan.navigationIDs = parentZone
|
|
769
|
+
? parentZone.sweepIDs
|
|
770
|
+
: zone.sweepIDs;
|
|
771
|
+
}
|
|
772
|
+
return calibratedPlan || null;
|
|
773
|
+
}
|
|
774
|
+
transformPosition3DForNewCalibration(position) {
|
|
775
|
+
const positionX = this.calibrationPlan.nameXAxis === 'x' ? position.x : position.z;
|
|
776
|
+
const positionY = this.calibrationPlan.nameYAxis === 'x' ? position.x : position.z;
|
|
777
|
+
let x = this.calibrationPlan.offsetX + positionX * this.calibrationPlan.scaleX;
|
|
778
|
+
let y = this.calibrationPlan.offsetY + positionY * this.calibrationPlan.scaleY;
|
|
779
|
+
const newCoeffX = this.coeffPlanX / (this.imgPlan.width / this.calibrationPlan.imgWidth);
|
|
780
|
+
// console.log("newCoeffX", newCoeffX);
|
|
781
|
+
const newCoeffY = this.coeffPlanY / (this.imgPlan.height / this.calibrationPlan.imgHeight);
|
|
782
|
+
x = (x - this.calibrationPlan.offsetXPlan) / this.calibrationPlan.scalePlan;
|
|
783
|
+
y = (y - this.calibrationPlan.offsetYPlan) / this.calibrationPlan.scalePlan;
|
|
784
|
+
// console.log(x, y);
|
|
785
|
+
return { x, y, coeffX: newCoeffX, coeffY: newCoeffY };
|
|
786
|
+
}
|
|
787
|
+
}
|
|
788
|
+
PlanService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.2.7", ngImport: i0, type: PlanService, deps: [{ token: 'currentAPIService' }, { token: i1.ZoneService }, { token: i2.NavigationService }, { token: i3.ViewerService }, { token: i4.Router }, { token: i5.MatterportService }, { token: i6.BaseVisibilityService }, { token: i7.ContentService }], target: i0.ɵɵFactoryTarget.Injectable });
|
|
789
|
+
PlanService.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "13.2.7", ngImport: i0, type: PlanService, providedIn: 'root' });
|
|
790
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.2.7", ngImport: i0, type: PlanService, decorators: [{
|
|
791
|
+
type: Injectable,
|
|
792
|
+
args: [{
|
|
793
|
+
providedIn: 'root',
|
|
794
|
+
}]
|
|
795
|
+
}], ctorParameters: function () { return [{ type: undefined, decorators: [{
|
|
796
|
+
type: Inject,
|
|
797
|
+
args: ['currentAPIService']
|
|
798
|
+
}] }, { type: i1.ZoneService }, { type: i2.NavigationService }, { type: i3.ViewerService }, { type: i4.Router }, { type: i5.MatterportService }, { type: i6.BaseVisibilityService }, { type: i7.ContentService }]; } });
|
|
799
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"plan.service.js","sourceRoot":"","sources":["../../../../../../projects/ngx-smarterplan-core/src/lib/services/models/plan.service.ts"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable func-names */\r\n/* eslint-disable class-methods-use-this */\r\nimport { Inject, Injectable } from '@angular/core';\r\nimport { getDocument, GlobalWorkerOptions } from 'pdfjs-dist';\r\nimport panzoom from 'panzoom';\r\nimport videojs from \"video.js\";\r\nimport { Subject } from 'rxjs';\r\nimport { Router } from '@angular/router';\r\nimport {\r\n    CommentType,\r\n    FeatureType,\r\n    Plan,\r\n    PoiType,\r\n    Zone,\r\n} from '../../types.service';\r\nimport { BaseVisibilityService } from '../baseVisibility.service';\r\nimport { MatterportService } from '../matterport.service';\r\nimport { ViewerService } from '../viewer.service';\r\nimport { ZoneService } from './zone.service';\r\nimport { NavigationService } from './navigation.service';\r\nimport { getSignedFile, uploadFileToS3 } from '../s3.service';\r\nimport { BaseTagService } from '../tag.service';\r\nimport { Config } from '../../config';\r\nimport { getCoefficientsForImage } from '../zone-drawer.service';\r\nimport { getMetaForImage } from '../../helpers.service';\r\nimport { ContentService } from '../content.service';\r\n\r\n@Injectable({\r\n    providedIn: 'root',\r\n})\r\nexport class PlanService {\r\n    chosenPlan: Plan;\r\n\r\n    currentPlan: Plan;\r\n\r\n    currentSweep: string;\r\n\r\n    planFileCache: File;\r\n\r\n    planCanvas: any;\r\n\r\n    planDiv: HTMLElement | SVGElement;\r\n    planDivContent: HTMLElement;\r\n\r\n    panzoom: any;\r\n\r\n    isReady: boolean = false;\r\n\r\n    userPositionBtn: HTMLButtonElement;\r\n\r\n    coeffPlanX: number;\r\n\r\n    coeffPlanY: number;\r\n\r\n    calibrationPlan: any;\r\n\r\n    detailTagDiv: HTMLElement;\r\n\r\n    detailTagDivIsHover = false;\r\n\r\n    btnTagIsHover = false;\r\n\r\n    resizePlanSubscription = new Subject<boolean>();\r\n\r\n    imgPlan: HTMLImageElement;\r\n\r\n    lastTouchTime = 0;\r\n\r\n    delayDblTouch = 500;\r\n\r\n    lastRotation: number = 0;\r\n\r\n    htmlContentToInject: {\r\n        title: string;\r\n        content: string;\r\n        elementID: string;\r\n        y: number;\r\n        x: number;\r\n        tagIcon: string;\r\n        url: string;\r\n    }[] = [];\r\n\r\n    cache: {\r\n        equipIDs: string[];\r\n        ticketIDs: string[];\r\n        measurementsIDs: string[];\r\n        featureIDs: string[];\r\n        zoneID: string;\r\n        plan: Plan;\r\n        htmlContent: {\r\n            title: string;\r\n            content: string;\r\n            elementID: string;\r\n            y: number;\r\n            x: number;\r\n            tagIcon: string;\r\n            url: string;\r\n        }[];\r\n    };\r\n\r\n    focusMouseTagDiv: boolean = false;\r\n\r\n    API: any; //AWS\r\n\r\n    constructor(\r\n        @Inject('currentAPIService') apiInjected: any,\r\n        private zoneService: ZoneService,\r\n        private navigationService: NavigationService,\r\n        // tagService: BaseTagService,\r\n        private viewerService: ViewerService,\r\n        private router: Router,\r\n        private matterportService: MatterportService,\r\n        // private config: AppConfig,\r\n        private visibilityService: BaseVisibilityService,\r\n        private contentService: ContentService\r\n    ) {\r\n        this.API = apiInjected;\r\n        this.matterportService.currentCameraPose.subscribe((pose) => {\r\n            this.updateRotation(pose.rotation.y);\r\n        });\r\n    }\r\n\r\n    async createPlan(planInput: Plan): Promise<Plan> {\r\n        return this.API.__proto__.CreatePlan(planInput);\r\n    }\r\n\r\n    async deletePlan(plan: Plan): Promise<Plan> {\r\n        // await deleteFromS3(plan.annexe);\r\n        return this.API.__proto__.DeletePlan({ id: plan.id });\r\n    }\r\n\r\n    async getPlansForSpace(spaceID: string): Promise<Plan[]> {\r\n        return this.API.__proto__\r\n            .PlansBySpace(spaceID)\r\n            .then((response) => response.items);\r\n    }\r\n\r\n    async getPlansForZone(zoneID: string): Promise<Plan[]> {\r\n        return this.API.__proto__\r\n            .PlansByZone(zoneID)\r\n            .then((response) => response.items);\r\n    }\r\n\r\n    async getSingedPlansForSpace(spaceID: string): Promise<Plan[]> {\r\n        const plansFromDB = await this.getPlansForSpace(spaceID);\r\n        // console.log(plansFromDB);\r\n        const plans = [];\r\n        if (plansFromDB.length > 0) {\r\n            await Promise.all(\r\n                plansFromDB.map(async (plan) => {\r\n                    const planObject = { ...plan } as Plan;\r\n                    // annexe looks like visits/modelID/plans/file.extension\r\n                    const signed = await getSignedFile(plan.annexe);\r\n                    if (signed) {\r\n                        planObject.filepath = signed;\r\n                    }\r\n                    if (plan.annexe) {\r\n                        const [, modelID, , filenameWithExtension] = plan.annexe.split('/');\r\n                        const [, extention] = filenameWithExtension.split('.');\r\n                        planObject.model3d = modelID;\r\n                        planObject.extension = extention;\r\n                        plans.push(planObject);\r\n                    } else {\r\n                        console.log(`Error plan have not annexe => `);\r\n                        console.log(plan.id);\r\n                    }\r\n                })\r\n            );\r\n        }\r\n        return plans;\r\n    }\r\n\r\n    async getPlansWithZonesForSpace(spaceID: string): Promise<Plan[]> {\r\n        const plans = await this.getPlansForSpace(spaceID);\r\n        const plansZones = plans.filter((plan) => plan.zone && plan.calibration);\r\n        return plansZones;\r\n    }\r\n\r\n    setChosenPlan(chosenPlan: Plan) {\r\n        this.chosenPlan = chosenPlan;\r\n    }\r\n\r\n    getChosenPlan(): Plan {\r\n        return this.chosenPlan;\r\n    }\r\n\r\n    setPlanFileCache(planFileCache: File): Promise<void> {\r\n        this.planFileCache = planFileCache;\r\n        return Promise.resolve();\r\n    }\r\n\r\n    getPlanFileCache(): File {\r\n        return this.planFileCache;\r\n    }\r\n\r\n    async createPlanWithAnnexe(plan: Plan, file: File): Promise<Plan> {\r\n        const createdPlan = await this.createPlan(plan);\r\n        const model3D = await this.zoneService.getModel3DForZone(plan.zoneID);\r\n        const url = await uploadFileToS3(\r\n            `visits/${model3D}/plans/`,\r\n            file,\r\n            createdPlan.id\r\n        );\r\n        if (url) {\r\n            return this.API.__proto__.UpdatePlan({ id: createdPlan.id, annexe: url });\r\n        }\r\n        return createdPlan;\r\n    }\r\n\r\n    async updatePlan(plan: any): Promise<Plan> {\r\n        return this.API.__proto__.UpdatePlan(plan);\r\n    }\r\n\r\n    async updatePlanFile(): Promise<string> {\r\n        if (this.chosenPlan && this.planFileCache) {\r\n            const model3D = await this.zoneService.getModel3DForZone(\r\n                this.chosenPlan.zoneID\r\n            );\r\n            return uploadFileToS3(\r\n                `visits/${model3D}/plans/`,\r\n                this.planFileCache,\r\n                this.chosenPlan.id\r\n            );\r\n        }\r\n        return Promise.reject();\r\n    }\r\n\r\n    async setAllPlansForZoneNotCurrent(zoneID: string, currentPlanID: string) {\r\n        const plans = await this.API.__proto__.ListPlans({\r\n            zoneID: { eq: zoneID },\r\n        });\r\n        await Promise.all(\r\n            plans.items.map(async (plan) => {\r\n                if (plan.isCurrentForZone && plan.id !== currentPlanID) {\r\n                    await this.updatePlan({\r\n                        id: plan.id,\r\n                        isCurrentForZone: false,\r\n                    });\r\n                }\r\n            })\r\n        );\r\n    }\r\n\r\n    async configurePlan(\r\n        currentPlan: Plan,\r\n        divId: string = 'planDivPane',\r\n        divContentId: string = 'planDivPaneContent'\r\n    ): Promise<boolean> {\r\n        this.isReady = false;\r\n        if (!this.detailTagDiv) {\r\n            this.detailTagDiv = document.querySelector(\r\n                `#detailTagDiv`\r\n            ) as HTMLElement;\r\n        }\r\n        this.detailTagDiv.addEventListener('mouseenter', () => {\r\n            this.focusMouseTagDiv = true;\r\n        });\r\n        this.detailTagDiv.addEventListener('mouseleave', () => {\r\n            this.btnTagIsHover = false;\r\n            this.detailTagDiv.style.display = 'none';\r\n            this.focusMouseTagDiv = false;\r\n        });\r\n\r\n        this.calibrationPlan = JSON.parse(currentPlan.calibration);\r\n        this.currentPlan = currentPlan;\r\n        this.planDiv = document.querySelector(`#${divId}`) as HTMLElement;\r\n        this.planDivContent = document.querySelector(\r\n            `#${divContentId}`\r\n        ) as HTMLElement;\r\n\r\n        this.planDiv.addEventListener('touchstart', this.handleTouch.bind(this));\r\n        this.planDiv.addEventListener('dblclick', this.onDblClickPlan.bind(this));\r\n\r\n        if (!currentPlan.annexe.includes('pdf')) {\r\n            this.imgPlan = await getMetaForImage(currentPlan.filepath);\r\n            //Rect use offsetWidth : height, not actual boundingClientRect because will be modified by the zoom.\r\n            const rect = {\r\n                width: this.planDivContent.offsetWidth,\r\n                height: this.planDivContent.offsetHeight,\r\n            };\r\n            const { coeffX, coeffY } = getCoefficientsForImage(this.imgPlan, rect);\r\n            this.coeffPlanX = coeffX;\r\n            this.coeffPlanY = coeffY;\r\n\r\n            this.planDivContent.style.backgroundImage = `url(${currentPlan.filepath})`;\r\n        } else {\r\n            await this.drawPdf(currentPlan);\r\n        }\r\n        this.planDivContent.style.backgroundSize = 'contain';\r\n        this.planDivContent.style.backgroundRepeat = 'no-repeat';\r\n\r\n        this.panzoom = panzoom(this.planDiv, {\r\n            bounds: true,\r\n            boundsPadding: 0,\r\n            maxZoom: 4,\r\n            zoomDoubleClickSpeed: 1, //disables double click zoom\r\n            //initialZoom: 4\r\n        });\r\n        this.panzoom.zoomAbs(0, 0, 4);\r\n        setTimeout(() => {\r\n            this.updateRotation(this.lastRotation);\r\n        }, 200);\r\n        return true;\r\n    }\r\n\r\n    clearPlanImage() {\r\n        if (this.planDivContent) {\r\n            this.planDivContent.style.backgroundImage = null;\r\n        }\r\n    }\r\n\r\n    async drawPdf(currentPlan: Plan): Promise<void> {\r\n        return new Promise(async (resolve) => {\r\n            const canvas = document.createElement('canvas') as HTMLCanvasElement;\r\n\r\n            const context = canvas.getContext('2d') as CanvasRenderingContext2D;\r\n            const pdfjsWorker = await import('pdfjs-dist/build/pdf.worker.entry');\r\n\r\n            GlobalWorkerOptions.workerSrc = pdfjsWorker;\r\n            const pdf = await getDocument(currentPlan.filepath).promise;\r\n            const page = await pdf.getPage(1);\r\n\r\n            const viewPortParameters = { scale: 1.5 };\r\n            const viewport = page.getViewport(viewPortParameters);\r\n\r\n            canvas.height = viewport.height;\r\n            canvas.width = viewport.width;\r\n            const renderContext = {\r\n                canvasContext: context,\r\n                viewport,\r\n            };\r\n            const renderTask = page.render(renderContext).promise;\r\n            renderTask.then(async () => {\r\n                const imgUrl = canvas.toDataURL('image/png');\r\n                this.imgPlan = await getMetaForImage(imgUrl);\r\n                const rect = {\r\n                    width: this.planDivContent.offsetWidth,\r\n                    height: this.planDivContent.offsetHeight,\r\n                };\r\n                const { coeffX, coeffY } = getCoefficientsForImage(this.imgPlan, rect);\r\n                this.coeffPlanX = coeffX;\r\n                this.coeffPlanY = coeffY;\r\n                this.planDiv.style.backgroundImage = `url(${imgUrl})`;\r\n                resolve();\r\n            });\r\n        });\r\n    }\r\n\r\n    async drawElement(\r\n        element: any,\r\n        tagType: PoiType,\r\n        currentPlan: Plan,\r\n        tagService: BaseTagService,\r\n        config: Config,\r\n        sizeButton: number = 10\r\n    ) {\r\n        const [poi] = element.pois.items;\r\n        if (poi && poi.coordinate && currentPlan) {\r\n            const { zone } = currentPlan;\r\n            if (zone) {\r\n                const position = JSON.parse(poi.coordinate);\r\n                const button = document.createElement('button');\r\n                let elementTitle: string;\r\n                const url = tagService.getUrlForSeeDetails(element, tagType);\r\n                let tagIconImage: string;\r\n                switch (tagType) {\r\n                    case PoiType.TICKET: {\r\n                        tagIconImage = config.my_config.icon_ticket;\r\n                        break;\r\n                    }\r\n                    case PoiType.EQUIPMENT: {\r\n                        tagIconImage = config.my_config.icon_equipment;\r\n                        break;\r\n                    }\r\n                    case PoiType.MEASURE:\r\n                        tagIconImage = config.my_config.icon_measure;\r\n                        break;\r\n                    case PoiType.OBJECT3D:\r\n                        tagIconImage = config.my_config.icon_object3d;\r\n                        break;\r\n                    case PoiType.DATA:\r\n                        tagIconImage = config.my_config.icon_data;\r\n                        break;\r\n                    case PoiType.DESK:\r\n                        tagIconImage = config.my_config.icon_data;\r\n                        break;\r\n                    default:\r\n                        return;\r\n                }\r\n                if (poi.tagIcon) {\r\n                    const tagIcon = JSON.parse(poi.tagIcon);\r\n                    if (\r\n                        tagType === PoiType.DATA &&\r\n                        element.type === FeatureType.INDICATOR_TEMP\r\n                    ) {\r\n                        tagIcon.src = tagService.getIconTagImageForFeature(element, poi);\r\n                    }\r\n                    // poi.tagIcon = tagIcon;\r\n\r\n                    if (tagIcon.src) {\r\n                        const source = await getSignedFile(tagIcon.src);\r\n                        if (source) {\r\n                            tagIconImage = source;\r\n                        }\r\n                    }\r\n                }\r\n                this.styleButton(button, tagIconImage, sizeButton);\r\n                button.id = element.id;\r\n                this.planDivContent.append(button);\r\n                let x: number;\r\n                let y: number;\r\n                if (this.calibrationPlan.new) {\r\n                    // alert(\"new calibration plan: positioning not ready yet\");\r\n                    const position2D =\r\n                        this.transformPosition3DForNewCalibration(position);\r\n                    x = position2D.x;\r\n                    y = position2D.y;\r\n                    button.style.top = `${y / position2D.coeffY - 5}px`;\r\n                    button.style.left = `${x / position2D.coeffX - 5}px`;\r\n                } else {\r\n                    // to remove when all calibrations are new\r\n                    x =\r\n                        this.calibrationPlan.offsetX + position.x * this.calibrationPlan.x;\r\n                    y =\r\n                        this.calibrationPlan.offsetY + position.z * this.calibrationPlan.y;\r\n                    button.style.top = `${y / this.coeffPlanY - 5}px`;\r\n                    button.style.left = `${x / this.coeffPlanX - 5}px`;\r\n                }\r\n\r\n                const html = await tagService.getHtmlToInject(tagType, element);\r\n                if (html != '') {\r\n                    this.htmlContentToInject.push({\r\n                        elementID: element.id,\r\n                        title: '',\r\n                        content: html,\r\n                        y,\r\n                        x,\r\n                        tagIcon: tagIconImage,\r\n                        url,\r\n                    });\r\n                } else {\r\n                    // when we don't have html => case of EMBED Comment type\r\n                    const commentEmbed = element.comments?.items.find(\r\n                        (com) => com.type === CommentType.EMBED && com.shownInTag\r\n                    );\r\n                    if (commentEmbed) {\r\n                        this.htmlContentToInject.push({\r\n                            elementID: element.id,\r\n                            title: element.title,\r\n                            content: `<iframe src=${commentEmbed.externalLink} height=\"200px\" width=\"100%\"></iframe>`,\r\n                            y,\r\n                            x,\r\n                            tagIcon: tagIconImage,\r\n                            url,\r\n                        });\r\n                    }\r\n                }\r\n\r\n                this.addListenersToButton(button, url, tagService, element);\r\n                this.updateRotation(this.lastRotation);\r\n            }\r\n        }\r\n    }\r\n\r\n    async drawUserPosition(currentSweep = this.currentSweep, sizeButton: number = 10) {\r\n        this.currentSweep = currentSweep;\r\n        if (this.calibrationPlan && currentSweep) {\r\n            const position =\r\n                this.matterportService.getCurrentCameraPosition()?.position;\r\n            if (position) {\r\n                if (!this.userPositionBtn) {\r\n                    this.userPositionBtn = document.createElement('button');\r\n                    this.userPositionBtn.id = 'currentPosition';\r\n                    this.planDivContent.append(this.userPositionBtn);\r\n                    this.userPositionBtn.title = 'You are here';\r\n                    this.styleButton(\r\n                        this.userPositionBtn,\r\n                        'https://api.iconify.design/bx:bxs-user-circle.svg?color=green&height=17&width=17',\r\n                        sizeButton\r\n                    );\r\n                }\r\n                let x: number;\r\n                let y: number;\r\n                if (this.calibrationPlan.new) {\r\n                    const position2D =\r\n                        this.transformPosition3DForNewCalibration(position);\r\n                    x = position2D.x / position2D.coeffX - 5;\r\n                    y = position2D.y / position2D.coeffY - 5;\r\n                } else {\r\n                    x =\r\n                        (this.calibrationPlan.offsetX +\r\n                            position.x * this.calibrationPlan.x) /\r\n                        this.coeffPlanX;\r\n                    y =\r\n                        (this.calibrationPlan.offsetY +\r\n                            position.z * this.calibrationPlan.y) /\r\n                        this.coeffPlanY;\r\n                }\r\n                this.userPositionBtn.style.top = `${y}px`;\r\n                this.userPositionBtn.style.left = `${x}px`;\r\n                this.moveOnPoint({ x: x + 5, y: y + 5 });\r\n            }\r\n        }\r\n    }\r\n\r\n    /**\r\n     * Center the view of the plan on coordinates\r\n     * @param coordinate Coordinate on the plan\r\n     */\r\n    moveOnPoint(coordinate) {\r\n        if (this.planDiv) {\r\n            const scale = this.panzoom.getTransform().scale;\r\n            const moveX = this.planDiv.clientWidth / 2 - coordinate.x * scale;\r\n            const moveY = this.planDiv.clientHeight / 2 - coordinate.y * scale;\r\n            this.panzoom.moveTo(moveX, moveY);\r\n            this.planDivContent.style.transformOrigin = `${coordinate.x}px ${coordinate.y}px`;\r\n        }\r\n    }\r\n\r\n    /**\r\n     * Apply a rotation to the plan\r\n     * Buttons will remain in the correct rotation\r\n     * @param rotation\r\n     */\r\n    updateRotation(rotation) {\r\n        this.lastRotation = rotation;\r\n        if (this.planDivContent) {\r\n            this.planDivContent.style.transform = `rotate(${rotation}deg)`;\r\n            const buttons = this.planDivContent.querySelectorAll('button');\r\n            let rotationRegex = /(rotate\\(-?\\d*\\.?\\d+deg\\))/g;\r\n            // eslint-disable-next-line unicorn/no-array-for-each\r\n            buttons.forEach((button) => {\r\n                button.style.transform = button.style.transform.replace(rotationRegex, \"\") + ` rotate(${-rotation}deg)`;\r\n            });\r\n        }\r\n    }\r\n\r\n    removeCurrentPosition() {\r\n        const button = document.querySelector('#currentPosition');\r\n        if (button) {\r\n            button.remove();\r\n            this.userPositionBtn = null;\r\n        }\r\n    }\r\n\r\n    styleButton(button: HTMLButtonElement, url: string, sizeButton: number = 10) {\r\n        button.style.backgroundImage = `url(${url})`;\r\n        button.style.backgroundColor = 'transparent';\r\n        button.style.position = 'absolute';\r\n        button.style.border = 'none';\r\n        button.style.width = `10px`;\r\n        button.style.height = `10px`;\r\n        button.disabled = false;\r\n        button.style.backgroundSize = 'contain';\r\n        button.style.backgroundRepeat = 'no-repeat';\r\n        button.style.transform = `scale(${sizeButton/10})`\r\n  }\r\n\r\n    addListenersToButton(\r\n        button: HTMLButtonElement,\r\n        url: string,\r\n        tagService: BaseTagService = null,\r\n        element: any = null\r\n    ) {\r\n        let videoJsPlayer;\r\n        button.addEventListener('mouseenter', async (event) => {\r\n            const contentForButton = this.htmlContentToInject.find(\r\n                (object) => object.elementID === button.id\r\n            );\r\n            this.btnTagIsHover = true;\r\n            const title = document.querySelector(`#titleTagDiv`) as HTMLElement;\r\n            if (contentForButton.title) {\r\n                title.style.display = 'inline';\r\n                title.innerHTML = contentForButton.title;\r\n            } else {\r\n                title.style.display = 'none';\r\n            }\r\n\r\n            const content = document.querySelector(`#htmlTagDiv`) as HTMLElement;\r\n            this.detailTagDiv.style.top = `${event.clientY + 10}px`;\r\n            this.detailTagDiv.style.left = `${event.clientX < 120 ? 0 : event.clientX - 120\r\n                }px`;\r\n            this.detailTagDiv.style.display = 'block';\r\n            // console.log(contentForButton.content);\r\n            content.innerHTML = contentForButton.content;\r\n            const detailButton = content.querySelector('#detailBtn');\r\n            if (detailButton) {\r\n                detailButton.addEventListener('click', () => {\r\n                    this.visibilityService.detailShowing.next(true);\r\n                    this.router.navigate([url]);\r\n                });\r\n            }\r\n\r\n            // handle image carousel fullscreen button\r\n            const imageFullscreenBtn = content.querySelector('#image-footer');\r\n            if (imageFullscreenBtn) {\r\n                imageFullscreenBtn.addEventListener('click', function () {\r\n                    if (tagService) {\r\n                        tagService.onActionImageClick(contentForButton.elementID);\r\n                    }\r\n                });\r\n            }\r\n\r\n            // handle booking fullscreen button\r\n            const bookingButton = content.querySelector('#bookingBtn');\r\n            if (bookingButton) {\r\n                bookingButton.addEventListener('click', () => {\r\n                    this.visibilityService.detailShowing.next(true);\r\n                    this.router.navigate([url]);\r\n                });\r\n            }\r\n\r\n            // handle document pdf fullscreen button\r\n            const docFullscreenBtn = content.querySelector('#doc-footer');\r\n            if (docFullscreenBtn && element) {\r\n                const documentUrl = tagService.getAnnexeForCommentTypeInFeature(\r\n                    element,\r\n                    CommentType.DOCUMENT\r\n                );\r\n                docFullscreenBtn.addEventListener('click', function () {\r\n                    if (tagService) {\r\n                        tagService.onActionDocClick(documentUrl);\r\n                    }\r\n                });\r\n            }\r\n\r\n            // handle video fullscreen button\r\n            const videoFullscreenBtn = content.querySelector('#btn-video-fullscreen');\r\n            if (videoFullscreenBtn && element) {\r\n                const videoUrl = tagService.getAnnexeForCommentTypeInFeature(\r\n                    element,\r\n                    CommentType.VIDEO\r\n                );\r\n                videoFullscreenBtn.addEventListener('click', function () {\r\n                    if (tagService) {\r\n                        tagService.onActionVideoClick(videoUrl);\r\n                    }\r\n                });\r\n            }\r\n\r\n            // handle Youtube video fullscreen button\r\n            const youtubeFullScreen = content.querySelector(\r\n                '#btn-video-fullscreen-youtube'\r\n            );\r\n            if (youtubeFullScreen && element) {\r\n                const youtubeUrl = tagService.getAnnexeForCommentTypeInFeature(\r\n                    element,\r\n                    CommentType.YOUTUBE\r\n                );\r\n                youtubeFullScreen.addEventListener('click', function () {\r\n                    if (tagService) {\r\n                        tagService.onActionYoutubeClick(youtubeUrl);\r\n                    }\r\n                });\r\n            }\r\n\r\n            // handle mute video button\r\n            const btnMute = document.getElementById('btn-mute');\r\n            const imgOn = document.getElementById('sound-on');\r\n            const imgOff = document.getElementById('sound-off');\r\n            const playerElement = document.getElementById(\"mus-video-tag\");\r\n\r\n            if (playerElement) {\r\n                videoJsPlayer = videojs(\"mus-video-tag\");\r\n                if (btnMute && videoJsPlayer) {\r\n                    btnMute.addEventListener('click', function () {\r\n                        if (videoJsPlayer.muted()) {\r\n                            videoJsPlayer.muted(false);\r\n                            if (imgOn && imgOff) {\r\n                                imgOn.style.display = 'inline-block';\r\n                                imgOff.style.display = 'none';\r\n                            }\r\n                        } else {\r\n                            videoJsPlayer.muted(true);\r\n                            if (imgOn && imgOff) {\r\n                                imgOn.style.display = 'none';\r\n                                imgOff.style.display = 'inline-block';\r\n                            }\r\n                        }\r\n                    });\r\n                }\r\n            }\r\n\r\n\r\n            // handle audio player buttons\r\n            const playerButton = document.querySelector('.audio-play-button');\r\n            const audio = document.getElementById('audio-tag') as HTMLMediaElement;\r\n            const timeline = document.querySelector(\r\n                '.timeline-tag'\r\n            ) as HTMLInputElement;\r\n            // const timelineValue = document.querySelector('.timeline-tag').value;\r\n            const soundButton = document.querySelector('.audio-sound-btn');\r\n            const playAudio = document.getElementById('play');\r\n            const pauseAudio = document.getElementById('pause');\r\n            const imgSoundOn = document.getElementById('audio-sound-on');\r\n            const imgSoundOff = document.getElementById('audio-sound-off');\r\n            const audioModal = document.getElementById('btn-audio-modal');\r\n            if (audio && playerButton && timeline) {\r\n                playerButton.addEventListener('click', function () {\r\n                    if (audio.paused) {\r\n                        audio.play();\r\n                        pauseAudio.style.display = 'inline-block';\r\n                        playAudio.style.display = 'none';\r\n                    } else {\r\n                        audio.pause();\r\n                        pauseAudio.style.display = 'none';\r\n                        playAudio.style.display = 'inline-block';\r\n                    }\r\n                });\r\n                soundButton.addEventListener('click', function () {\r\n                    if (audio.muted) {\r\n                        audio.muted = false;\r\n                        imgSoundOn.style.display = 'inline-block';\r\n                        imgSoundOff.style.display = 'none';\r\n                    } else {\r\n                        audio.muted = true;\r\n                        imgSoundOn.style.display = 'none';\r\n                        imgSoundOff.style.display = 'inline-block';\r\n                    }\r\n                });\r\n                function changeTimelinePosition() {\r\n                    if (audio.duration) {\r\n                        const percentagePosition =\r\n                            (100 * audio.currentTime) / audio.duration;\r\n                        const percentagePositionString = percentagePosition + '%';\r\n                        timeline.style.backgroundSize = percentagePositionString + ' 100%';\r\n                        timeline.value = percentagePosition.toString();\r\n                    }\r\n                }\r\n                audio.ontimeupdate = changeTimelinePosition;\r\n                function changeSeek() {\r\n                    const time = (parseInt(timeline.value) * audio.duration) / 100;\r\n                    audio.currentTime = time;\r\n                }\r\n                timeline.addEventListener('change', changeSeek);\r\n                timeline.addEventListener('input', changeSeek);\r\n                // handle audio click\r\n                if (audioModal && element) {\r\n                    const audioComment = element.comments?.items.find(\r\n                        (com) => com.type === CommentType.AUDIO && com.shownInTag\r\n                    );\r\n                    const audioCommentID = audioComment ? audioComment.id : '';\r\n                    audioModal.addEventListener('click', function () {\r\n                        if (tagService) {\r\n                            tagService.onActionAudioClick(audioCommentID);\r\n                        }\r\n                    });\r\n                }\r\n            }\r\n        });\r\n\r\n        button.addEventListener('mouseleave', async () => {\r\n            // we wait if user hovers on to div of the tag (changes this.focusMouseTagDiv)\r\n            this.btnTagIsHover = false;\r\n            setTimeout(() => {\r\n                if (!this.focusMouseTagDiv && !this.btnTagIsHover) {\r\n                    this.detailTagDiv.style.display = 'none';\r\n                    if (videoJsPlayer) {\r\n                        videoJsPlayer.dispose();\r\n                    }\r\n                }\r\n            }, 100);\r\n        });\r\n        button.addEventListener('click', async () => {\r\n            this.detailTagDiv.style.display = 'none';\r\n            if (videoJsPlayer) {\r\n                videoJsPlayer.dispose();\r\n            }\r\n            await this.viewerService.action_move_to_tag(button.id);\r\n            this.visibilityService.detailShowing.next(true);\r\n            this.router.navigate([url]);\r\n        });\r\n    }\r\n\r\n    clearBtn(idList: any[]) {\r\n        for (const id of idList) {\r\n            const button = document.getElementById(id);\r\n            if (button) {\r\n                button.remove();\r\n            }\r\n        }\r\n    }\r\n\r\n    clearAllButtons() {\r\n        if (this.planDivContent) {\r\n            const buttons = this.planDivContent.querySelectorAll('button');\r\n            // eslint-disable-next-line unicorn/no-array-for-each\r\n            buttons.forEach(function (currentValue) {\r\n                currentValue.remove();\r\n            });\r\n            this.userPositionBtn = null;\r\n        }\r\n    }\r\n\r\n    updateAllButtonsStyle(properties: Array<string>, values: Array<string>) {\r\n        if (this.planDivContent) {\r\n            const buttons = this.planDivContent.querySelectorAll('button');\r\n            // eslint-disable-next-line unicorn/no-array-for-each\r\n            buttons.forEach(function (currentValue) {\r\n                properties.forEach((property, i) => {\r\n                    if (property === \"transform\") {\r\n                        //For saving the rotation and adding scale\r\n                        let rotationRegex = /(rotate\\(-?\\d*\\.?\\d+deg\\))/g;\r\n                        let rotationMatch = currentValue.style[property].match(rotationRegex);\r\n                        if (rotationMatch.length > 0) {\r\n                            currentValue.style[property] = `${values[i]} ${rotationMatch[0]}`;\r\n                        }\r\n                    } else {\r\n                        currentValue.style[property] = values[i];\r\n                    }\r\n\r\n                });\r\n            });\r\n        }\r\n    }\r\n\r\n    onPlanRemove() {\r\n        this.userPositionBtn = null;\r\n        this.calibrationPlan = null;\r\n        this.currentSweep = null;\r\n        this.detailTagDiv = null;\r\n    }\r\n\r\n    async resizePlan(isRemoving: boolean) {\r\n        this.resizePlanSubscription.next(false);\r\n        if (isRemoving) {\r\n            return;\r\n        }\r\n        console.log('in resizePlan');\r\n        if (this.planDiv && this.currentSweep) {\r\n            this.panzoom = panzoom(this.planDiv, {\r\n                bounds: true,\r\n                boundsPadding: 0,\r\n                maxZoom: 4,\r\n            });\r\n            setTimeout(() => {\r\n                const rect = {\r\n                    width: this.planDivContent.offsetWidth,\r\n                    height: this.planDivContent.offsetHeight,\r\n                };\r\n                const { coeffX, coeffY } = getCoefficientsForImage(this.imgPlan, rect);\r\n                this.coeffPlanX = coeffX;\r\n                this.coeffPlanY = coeffY;\r\n                this.panzoom.zoomAbs(0, 0, 4);\r\n                this.resizePlanSubscription.next(true);\r\n            }, 200);\r\n        }\r\n    }\r\n\r\n    handleTouch(event) {\r\n        const now = Date.now();\r\n        if (now - this.lastTouchTime < this.delayDblTouch) {\r\n            this.lastTouchTime = 0;\r\n            // this.onDblClickPlan(event);\r\n            /** Simulate a dblclick for phone (we don't get offsetX/Y on touchevent) */\r\n            const newEvent = document.createEvent('MouseEvents');\r\n            const touch = event.changedTouches[0];\r\n            newEvent.initMouseEvent(\r\n                'dblclick',\r\n                true,\r\n                true,\r\n                event.target.ownerDocument.defaultView,\r\n                0,\r\n                touch.screenX,\r\n                touch.screenY,\r\n                touch.clientX,\r\n                touch.clientY,\r\n                event.ctrlKey,\r\n                event.altKey,\r\n                event.shirtKey,\r\n                event.metaKey,\r\n                0,\r\n                null\r\n            );\r\n            event.target.dispatchEvent(newEvent);\r\n        } else {\r\n            this.lastTouchTime = now;\r\n        }\r\n    }\r\n\r\n    async onDblClickPlan(event) {\r\n        event.preventDefault();\r\n        const clickX = event.offsetX;\r\n        const clickY = event.offsetY;\r\n        const zonePlan = this.currentPlan.zone;\r\n        if (zonePlan) {\r\n            const navigations: any =\r\n                await this.navigationService.getNavigationsForZone(zonePlan);\r\n            if (navigations.length > 0) {\r\n                await Promise.all(\r\n                    navigations.map(async (nav) => {\r\n                        const position = JSON.parse(nav.position);\r\n                        let x: number;\r\n                        let y: number;\r\n                        if (this.calibrationPlan.new) {\r\n                            const position2D =\r\n                                this.transformPosition3DForNewCalibration(position);\r\n                            x = position2D.x / position2D.coeffX;\r\n                            y = position2D.y / position2D.coeffY;\r\n                        } else {\r\n                            x =\r\n                                (this.calibrationPlan.offsetX +\r\n                                    position.x * this.calibrationPlan.x) /\r\n                                this.coeffPlanX;\r\n                            y =\r\n                                (this.calibrationPlan.offsetY +\r\n                                    position.z * this.calibrationPlan.y) /\r\n                                this.coeffPlanY;\r\n                        }\r\n\r\n                        const distX = Math.abs(clickX - x);\r\n                        const distY = Math.abs(clickY - y);\r\n                        nav.dist = Math.sqrt(Math.pow(distX, 2) + Math.pow(distY, 2));\r\n                    })\r\n                );\r\n                navigations.sort((a, b) => {\r\n                    return a.dist - b.dist;\r\n                });\r\n                this.matterportService.action_go_to_sweep(\r\n                    navigations[0].matterportSweepID\r\n                );\r\n            }\r\n        }\r\n    }\r\n\r\n    /**\r\n     * Configures plan from cache (previous plan).\r\n     */\r\n    async uploadPlanFromCache(): Promise<void> {\r\n        this.htmlContentToInject = this.cache.htmlContent;\r\n        await this.configurePlan(this.cache.plan);\r\n        return Promise.resolve();\r\n    }\r\n\r\n    /**\r\n     * Puts buttons (tags) from cache according to passed element IDs (for filter if any)\r\n     * @param elementIDs elements (tickets, equipments, etc) to be shown on plan\r\n     */\r\n    uploadTagsFromCache(sizeButton: number = 10) {\r\n        // console.log(\"uploading tags from cache\");\r\n\r\n        for (const cached of this.cache.htmlContent) {\r\n            const button = document.createElement('button');\r\n            this.styleButton(button, cached.tagIcon, sizeButton);\r\n            button.id = cached.elementID;\r\n            this.planDivContent.append(button);\r\n            if (this.calibrationPlan.new) {\r\n                const newCoeffX =\r\n                    this.coeffPlanX /\r\n                    (this.imgPlan.width / this.calibrationPlan.imgWidth);\r\n                const newCoeffY =\r\n                    this.coeffPlanY /\r\n                    (this.imgPlan.height / this.calibrationPlan.imgHeight);\r\n                button.style.top = `${cached.y / newCoeffY - 5}px`;\r\n                button.style.left = `${cached.x / newCoeffX - 5}px`;\r\n            } else {\r\n                button.style.top = `${cached.y / this.coeffPlanY - 5}px`;\r\n                button.style.left = `${cached.x / this.coeffPlanX - 5}px`;\r\n            }\r\n\r\n            this.addListenersToButton(button, cached.url);\r\n        }\r\n    }\r\n\r\n    async getCalibratedPlanForZone(zone: Zone): Promise<Plan> {\r\n        let plans = (await this.API.PlansByZone(zone.id)).items;\r\n        let parentZone = null;\r\n        if (plans.length === 0) {\r\n            // try to find parent zone\r\n            parentZone = await this.API.GetZone(zone.parentID);\r\n            plans = (await this.API.PlansByZone(zone.parentID)).items;\r\n        }\r\n        const calibratedPlan = plans.find(\r\n            (plan) => plan.calibration && plan.isCurrentForZone\r\n        ) as Plan;\r\n        if (calibratedPlan) {\r\n            const signed = await getSignedFile(calibratedPlan.annexe);\r\n            if (signed) {\r\n                calibratedPlan.filepath = signed;\r\n            }\r\n            calibratedPlan.navigationIDs = parentZone\r\n                ? parentZone.sweepIDs\r\n                : zone.sweepIDs;\r\n        }\r\n        return calibratedPlan || null;\r\n    }\r\n\r\n    transformPosition3DForNewCalibration(position: {\r\n        x: number;\r\n        y: number;\r\n        z: number;\r\n    }): { x: number; y: number; coeffX: number; coeffY: number; } {\r\n        const positionX =\r\n            this.calibrationPlan.nameXAxis === 'x' ? position.x : position.z;\r\n        const positionY =\r\n            this.calibrationPlan.nameYAxis === 'x' ? position.x : position.z;\r\n        let x =\r\n            this.calibrationPlan.offsetX + positionX * this.calibrationPlan.scaleX;\r\n        let y =\r\n            this.calibrationPlan.offsetY + positionY * this.calibrationPlan.scaleY;\r\n        const newCoeffX =\r\n            this.coeffPlanX / (this.imgPlan.width / this.calibrationPlan.imgWidth);\r\n        // console.log(\"newCoeffX\", newCoeffX);\r\n        const newCoeffY =\r\n            this.coeffPlanY / (this.imgPlan.height / this.calibrationPlan.imgHeight);\r\n        x = (x - this.calibrationPlan.offsetXPlan) / this.calibrationPlan.scalePlan;\r\n\r\n        y = (y - this.calibrationPlan.offsetYPlan) / this.calibrationPlan.scalePlan;\r\n        // console.log(x, y);\r\n\r\n        return { x, y, coeffX: newCoeffX, coeffY: newCoeffY };\r\n    }\r\n}\r\n"]}
|