@smarterplan/ngx-smarterplan-core 1.4.5 → 1.4.7
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/esm2022/lib/helpers.service.mjs +27 -20
- package/esm2022/lib/mattertagData.mjs +84 -84
- package/esm2022/lib/pipes/time-date-to-local-string.pipe.mjs +3 -8
- package/esm2022/lib/services/locale.service.mjs +3 -5
- package/esm2022/lib/services/matterport-import.service.mjs +3 -3
- package/esm2022/lib/services/matterport-measurement.service.mjs +78 -0
- package/esm2022/lib/services/matterport-navigation.service.mjs +157 -0
- package/esm2022/lib/services/matterport-object3d.service.mjs +234 -0
- package/esm2022/lib/services/matterport-pointer.service.mjs +130 -0
- package/esm2022/lib/services/matterport-tag.service.mjs +458 -0
- package/esm2022/lib/services/matterport.service.mjs +427 -988
- package/esm2022/lib/services/models/equipment.service.mjs +3 -4
- package/esm2022/lib/services/models/feature.service.mjs +3 -3
- package/esm2022/lib/services/models/measurement.service.mjs +3 -3
- package/esm2022/lib/services/models/ticket.service.mjs +3 -4
- package/esm2022/lib/services/navigator.service.mjs +10 -2
- package/esm2022/lib/services/tag.service.mjs +45 -19
- package/esm2022/lib/services/viewer.service.mjs +51 -16
- package/esm2022/lib/types.service.mjs +1 -1
- package/fesm2022/smarterplan-ngx-smarterplan-core.mjs +2028 -1490
- package/fesm2022/smarterplan-ngx-smarterplan-core.mjs.map +1 -1
- package/lib/helpers.service.d.ts +6 -0
- package/lib/mattertagData.d.ts +52 -10
- package/lib/services/matterport-measurement.service.d.ts +35 -0
- package/lib/services/matterport-navigation.service.d.ts +38 -0
- package/lib/services/matterport-object3d.service.d.ts +31 -0
- package/lib/services/matterport-pointer.service.d.ts +31 -0
- package/lib/services/matterport-tag.service.d.ts +104 -0
- package/lib/services/matterport.service.d.ts +91 -92
- package/lib/services/models/equipment.service.d.ts +2 -2
- package/lib/services/models/feature.service.d.ts +2 -2
- package/lib/services/models/measurement.service.d.ts +2 -2
- package/lib/services/models/ticket.service.d.ts +2 -2
- package/lib/services/tag.service.d.ts +4 -0
- package/lib/services/viewer.service.d.ts +1 -1
- package/lib/types.service.d.ts +2 -0
- package/package.json +21 -1
- package/esm2022/lib/matterport-extensions/hsl-loader/HlsLoader.mjs +0 -66
- package/esm2022/lib/matterport-extensions/video-renderer/VideoRenderer.mjs +0 -63
- package/lib/matterport-extensions/hsl-loader/HlsLoader.d.ts +0 -26
- package/lib/matterport-extensions/video-renderer/VideoRenderer.d.ts +0 -26
|
@@ -0,0 +1,458 @@
|
|
|
1
|
+
import { isDevMode, Injectable } from '@angular/core';
|
|
2
|
+
import { Subject } from 'rxjs';
|
|
3
|
+
import { MattertagData } from '../mattertagData';
|
|
4
|
+
import { TagAction, CameraMode } from '../types.service';
|
|
5
|
+
import { wait } from '../helpers.service';
|
|
6
|
+
import * as i0 from "@angular/core";
|
|
7
|
+
import * as i1 from "./matterport-navigation.service";
|
|
8
|
+
export class MatterportTagService {
|
|
9
|
+
navigationService;
|
|
10
|
+
dictionnaryTags = new Map();
|
|
11
|
+
tagAddQueue = Promise.resolve();
|
|
12
|
+
mattertagToFollow = null;
|
|
13
|
+
onGoToTag = new Subject();
|
|
14
|
+
mattertagIDs = [];
|
|
15
|
+
tagsAttachments = {};
|
|
16
|
+
tagMessengerOn = false;
|
|
17
|
+
tagService = null;
|
|
18
|
+
tagsCurrentOpacity = new Map();
|
|
19
|
+
tagsCurrentEnabled = new Map();
|
|
20
|
+
constructor(navigationService) {
|
|
21
|
+
this.navigationService = navigationService;
|
|
22
|
+
}
|
|
23
|
+
setTagService(service) {
|
|
24
|
+
this.tagService = service;
|
|
25
|
+
}
|
|
26
|
+
/** Opens a specific tag in the Matterport viewer by its ID. */
|
|
27
|
+
async goToTag(sdk, sid) {
|
|
28
|
+
if (!sdk)
|
|
29
|
+
return;
|
|
30
|
+
try {
|
|
31
|
+
this.onGoToTag.next(sid);
|
|
32
|
+
await sdk.Tag.open(sid);
|
|
33
|
+
}
|
|
34
|
+
catch (e) {
|
|
35
|
+
if (isDevMode())
|
|
36
|
+
console.warn('Cannot open tag', sid, e);
|
|
37
|
+
}
|
|
38
|
+
}
|
|
39
|
+
/** Opens the most recently created tag in the viewer. */
|
|
40
|
+
async goToLastTag(sdk) {
|
|
41
|
+
if (!sdk || this.mattertagIDs.length === 0)
|
|
42
|
+
return;
|
|
43
|
+
const lastSid = this.mattertagIDs[this.mattertagIDs.length - 1];
|
|
44
|
+
return this.goToTag(sdk, lastSid);
|
|
45
|
+
}
|
|
46
|
+
/**
|
|
47
|
+
* Adds a Mattertag to the viewer, serializing calls to prevent SDK race conditions.
|
|
48
|
+
* Reuses existing IDs if the tag is already present in the session.
|
|
49
|
+
*/
|
|
50
|
+
async addMattertagToViewer(sdk, mattertagData) {
|
|
51
|
+
if (!sdk)
|
|
52
|
+
return null;
|
|
53
|
+
this.tagAddQueue = this.tagAddQueue.then(async () => {
|
|
54
|
+
const data = mattertagData.getData();
|
|
55
|
+
const existingId = data?.id || null;
|
|
56
|
+
try {
|
|
57
|
+
const sidList = await sdk.Tag.add(data);
|
|
58
|
+
if (sidList && sidList.length > 0) {
|
|
59
|
+
const mattertagID = sidList[0];
|
|
60
|
+
this.mattertagIDs.push(mattertagID);
|
|
61
|
+
this.dictionnaryTags.set(mattertagID, mattertagData);
|
|
62
|
+
return mattertagID;
|
|
63
|
+
}
|
|
64
|
+
}
|
|
65
|
+
catch (error) {
|
|
66
|
+
if (error?.message?.includes('already in use') && existingId) {
|
|
67
|
+
if (!this.dictionnaryTags.has(existingId)) {
|
|
68
|
+
this.mattertagIDs.push(existingId);
|
|
69
|
+
this.dictionnaryTags.set(existingId, mattertagData);
|
|
70
|
+
}
|
|
71
|
+
return existingId;
|
|
72
|
+
}
|
|
73
|
+
if (isDevMode())
|
|
74
|
+
console.warn('Failed to add tag to viewer', error);
|
|
75
|
+
}
|
|
76
|
+
return null;
|
|
77
|
+
});
|
|
78
|
+
return this.tagAddQueue;
|
|
79
|
+
}
|
|
80
|
+
/**
|
|
81
|
+
* Internal method to add and configure a Mattertag with icon, opacity, and HTML content.
|
|
82
|
+
* Handles existing tags gracefully and queues operations to avoid SDK conflicts.
|
|
83
|
+
*/
|
|
84
|
+
async _doAddMattertag(sdk, mattertagData, setTagIconAndOpacity, injectHtmlInTag) {
|
|
85
|
+
const descriptor = mattertagData.getMattertagDescriptor();
|
|
86
|
+
//console.log('descriptor', descriptor);
|
|
87
|
+
const sid = descriptor[0]['id'];
|
|
88
|
+
if (sid && this.dictionnaryTags.has(sid)) {
|
|
89
|
+
await setTagIconAndOpacity(sid, mattertagData);
|
|
90
|
+
await injectHtmlInTag(mattertagData.getType(), mattertagData.getObject(), sid);
|
|
91
|
+
return sid;
|
|
92
|
+
}
|
|
93
|
+
this.tagAddQueue = this.tagAddQueue.then(async () => {
|
|
94
|
+
try {
|
|
95
|
+
const [createdSid] = await sdk.Tag.add(...descriptor);
|
|
96
|
+
this.mattertagIDs.push(createdSid);
|
|
97
|
+
this.dictionnaryTags.set(createdSid, mattertagData);
|
|
98
|
+
await wait(100);
|
|
99
|
+
await setTagIconAndOpacity(createdSid, mattertagData);
|
|
100
|
+
injectHtmlInTag(mattertagData.getType(), mattertagData.getObject(), createdSid).catch(err => {
|
|
101
|
+
if (isDevMode())
|
|
102
|
+
console.warn('HTML injection failed for', createdSid, err);
|
|
103
|
+
});
|
|
104
|
+
return createdSid;
|
|
105
|
+
}
|
|
106
|
+
catch (e) {
|
|
107
|
+
if (e?.message?.includes('already in use') && sid) {
|
|
108
|
+
if (!this.dictionnaryTags.has(sid)) {
|
|
109
|
+
this.mattertagIDs.push(sid);
|
|
110
|
+
this.dictionnaryTags.set(sid, mattertagData);
|
|
111
|
+
}
|
|
112
|
+
await setTagIconAndOpacity(sid, mattertagData);
|
|
113
|
+
await injectHtmlInTag(mattertagData.getType(), mattertagData.getObject(), sid);
|
|
114
|
+
return sid;
|
|
115
|
+
}
|
|
116
|
+
throw e;
|
|
117
|
+
}
|
|
118
|
+
});
|
|
119
|
+
return this.tagAddQueue;
|
|
120
|
+
}
|
|
121
|
+
getDistance(p1, p2) {
|
|
122
|
+
return Math.sqrt(Math.pow(p2.x - p1.x, 2) + Math.pow(p2.y - p1.y, 2) + Math.pow(p2.z - p1.z, 2));
|
|
123
|
+
}
|
|
124
|
+
/**
|
|
125
|
+
* Checks if a tag is within the bounding box of any active room (adding a slight margin to account for wall-mounted tags).
|
|
126
|
+
* If there are no active rooms (e.g., hallway), falls back to returning true.
|
|
127
|
+
*/
|
|
128
|
+
isTagInRooms(mattertagData, rooms) {
|
|
129
|
+
if (!rooms || rooms.length === 0)
|
|
130
|
+
return false;
|
|
131
|
+
const tagPos = mattertagData.getPosition();
|
|
132
|
+
const margin = 1.5;
|
|
133
|
+
return rooms.some((room) => {
|
|
134
|
+
if (!room.bounds)
|
|
135
|
+
return false;
|
|
136
|
+
const { min, max } = room.bounds;
|
|
137
|
+
return (tagPos.x >= min.x - margin && tagPos.x <= max.x + margin &&
|
|
138
|
+
tagPos.y >= min.y - margin && tagPos.y <= max.y + margin &&
|
|
139
|
+
tagPos.z >= min.z - margin && tagPos.z <= max.z + margin);
|
|
140
|
+
});
|
|
141
|
+
}
|
|
142
|
+
/**
|
|
143
|
+
* Computes the final context-aware opacity for a single tag.
|
|
144
|
+
* @param currentFloorSequence Authoritative current floor sequence from sdk.Floor.current
|
|
145
|
+
* (null = single-floor model or unknown).
|
|
146
|
+
* @param currentRooms Array of active rooms the camera is in.
|
|
147
|
+
*/
|
|
148
|
+
/*computeSingleTagContextOpacity(
|
|
149
|
+
mattertagData: MattertagData,
|
|
150
|
+
poseCamera: any,
|
|
151
|
+
sweepCollection: any,
|
|
152
|
+
currentFloorSequence: number | null,
|
|
153
|
+
currentRooms: any[]
|
|
154
|
+
): number {
|
|
155
|
+
const mode = this.navigationService.currentCameraMode;
|
|
156
|
+
|
|
157
|
+
// 1. Check Floor
|
|
158
|
+
const tagFloorId = this.getFloorIdForTag(mattertagData, sweepCollection);
|
|
159
|
+
const currentFloorId = currentFloorSequence?.toString();
|
|
160
|
+
if (tagFloorId && currentFloorId && tagFloorId !== currentFloorId) {
|
|
161
|
+
return 0;
|
|
162
|
+
}
|
|
163
|
+
|
|
164
|
+
// 2. Si Dollhouse -> Pleine opacité
|
|
165
|
+
//if (mode !== CameraMode.INSIDE) return 1;
|
|
166
|
+
|
|
167
|
+
// 3. Si Inside -> Check pièce + Distance progressive
|
|
168
|
+
//if (!this.isTagInRooms(mattertagData, currentRooms)) return 0;
|
|
169
|
+
|
|
170
|
+
return 1;
|
|
171
|
+
}*/
|
|
172
|
+
/** Returns the ID of the most recently added tag. */
|
|
173
|
+
getLastTag() {
|
|
174
|
+
if (this.mattertagIDs.length === 0)
|
|
175
|
+
return null;
|
|
176
|
+
return this.mattertagIDs[this.mattertagIDs.length - 1];
|
|
177
|
+
}
|
|
178
|
+
/**
|
|
179
|
+
* Updates the visibility (opacity) of all tags based on the current camera mode,
|
|
180
|
+
* floor and room context. Must be called after each sweep, floor or mode change.
|
|
181
|
+
*
|
|
182
|
+
* Rules:
|
|
183
|
+
* - DOLLHOUSE / FLOORPLAN → all tags hidden (opacity 0).
|
|
184
|
+
* - INSIDE → only tags whose sweep belongs to the current floor are shown;
|
|
185
|
+
* proximity-based opacity is applied via computeTagOpacity().
|
|
186
|
+
* Tags on other floors are hidden (opacity 0).
|
|
187
|
+
* - TRANSITIONING → no change (skip update to avoid flickering).
|
|
188
|
+
*
|
|
189
|
+
* @param sdk Matterport SDK instance.
|
|
190
|
+
* @param poseCamera Current camera pose.
|
|
191
|
+
* @param sweepCollection Full collection of sweeps.
|
|
192
|
+
* @param currentCameraMode Current camera mode.
|
|
193
|
+
* @param mattertagToFollow ID of the cursor tag (never touched).
|
|
194
|
+
*/
|
|
195
|
+
async updateTagsVisibilityForContext(sdk, sweepCollection, currentCameraMode, mattertagToFollow, currentFloorSequence = null, currentRooms = []) {
|
|
196
|
+
if (!sdk || this.dictionnaryTags.size === 0)
|
|
197
|
+
return;
|
|
198
|
+
const isInteriorMode = currentCameraMode === CameraMode.INSIDE;
|
|
199
|
+
const currentFloorId = currentFloorSequence !== null ? currentFloorSequence.toString() : null;
|
|
200
|
+
const promises = [];
|
|
201
|
+
for (const [tagId, mattertagData] of this.dictionnaryTags) {
|
|
202
|
+
if (tagId === mattertagToFollow)
|
|
203
|
+
continue;
|
|
204
|
+
const tagFloorId = this.getFloorIdForTag(mattertagData, sweepCollection);
|
|
205
|
+
const isOnCurrentFloor = !tagFloorId || !currentFloorId || tagFloorId === currentFloorId;
|
|
206
|
+
let targetOpacity = 0;
|
|
207
|
+
let shouldBeEnabled = false;
|
|
208
|
+
// Calculation of opacity and enable status
|
|
209
|
+
if (isOnCurrentFloor) {
|
|
210
|
+
if (!isInteriorMode) {
|
|
211
|
+
targetOpacity = 1;
|
|
212
|
+
shouldBeEnabled = true;
|
|
213
|
+
}
|
|
214
|
+
else {
|
|
215
|
+
targetOpacity = 0.9;
|
|
216
|
+
shouldBeEnabled = true;
|
|
217
|
+
}
|
|
218
|
+
}
|
|
219
|
+
// Apply changes
|
|
220
|
+
// Tooltip and click management
|
|
221
|
+
const lastEnabled = this.tagsCurrentEnabled.get(tagId);
|
|
222
|
+
if (lastEnabled !== shouldBeEnabled) {
|
|
223
|
+
this.tagsCurrentEnabled.set(tagId, shouldBeEnabled);
|
|
224
|
+
promises.push(sdk.Tag.allowAction(tagId, {
|
|
225
|
+
opening: shouldBeEnabled,
|
|
226
|
+
navigating: shouldBeEnabled,
|
|
227
|
+
docking: false
|
|
228
|
+
}).catch(() => { }));
|
|
229
|
+
}
|
|
230
|
+
// Opacity management
|
|
231
|
+
const lastOpacity = this.tagsCurrentOpacity.get(tagId) ?? -1;
|
|
232
|
+
if (Math.abs(lastOpacity - targetOpacity) > 0.05) {
|
|
233
|
+
this.tagsCurrentOpacity.set(tagId, targetOpacity);
|
|
234
|
+
promises.push(sdk.Tag.editOpacity(tagId, targetOpacity).catch(() => { }));
|
|
235
|
+
}
|
|
236
|
+
}
|
|
237
|
+
await Promise.all(promises);
|
|
238
|
+
}
|
|
239
|
+
/** Removes a specific tag from the viewer and local dictionaries. */
|
|
240
|
+
async deleteMattertagFromId(sdk, mattertagID) {
|
|
241
|
+
try {
|
|
242
|
+
if (sdk)
|
|
243
|
+
await sdk.Tag.remove(mattertagID);
|
|
244
|
+
this.dictionnaryTags.delete(mattertagID);
|
|
245
|
+
const index = this.mattertagIDs.indexOf(mattertagID);
|
|
246
|
+
if (index > -1)
|
|
247
|
+
this.mattertagIDs.splice(index, 1);
|
|
248
|
+
}
|
|
249
|
+
catch (e) {
|
|
250
|
+
if (isDevMode())
|
|
251
|
+
console.warn('Cannot delete tag', mattertagID, e);
|
|
252
|
+
}
|
|
253
|
+
}
|
|
254
|
+
/** Removes the last added tag from the viewer. */
|
|
255
|
+
async deleteLastMattertag(sdk) {
|
|
256
|
+
const lastSid = this.getLastTag();
|
|
257
|
+
if (lastSid)
|
|
258
|
+
await this.deleteMattertagFromId(sdk, lastSid);
|
|
259
|
+
}
|
|
260
|
+
/** Clears all tags from the viewer and resets local state. */
|
|
261
|
+
async action_delete_all_mattertags(sdk) {
|
|
262
|
+
if (sdk) {
|
|
263
|
+
const sidsToRemove = [...this.mattertagIDs];
|
|
264
|
+
for (const sid of sidsToRemove) {
|
|
265
|
+
try {
|
|
266
|
+
await sdk.Tag.remove(sid);
|
|
267
|
+
}
|
|
268
|
+
catch (e) { }
|
|
269
|
+
}
|
|
270
|
+
}
|
|
271
|
+
this.clear();
|
|
272
|
+
}
|
|
273
|
+
/**
|
|
274
|
+
* Creates and configures a Mattertag from a POI, handling coordinates, normals, and sweep ID.
|
|
275
|
+
* Skips creation if a tag for the object already exists.
|
|
276
|
+
*/
|
|
277
|
+
async createMattertagFromPOI(sdk, tagType, object, poi, poseCamera, getTagFromElementId, setTagIconAndOpacity, injectHtmlInTag) {
|
|
278
|
+
const { tag } = getTagFromElementId(object.id);
|
|
279
|
+
if (tag)
|
|
280
|
+
return;
|
|
281
|
+
const mattertagData = new MattertagData(tagType);
|
|
282
|
+
mattertagData.setObject(object, tagType);
|
|
283
|
+
if (poi.coordinate) {
|
|
284
|
+
try {
|
|
285
|
+
mattertagData.setPosition(JSON.parse(poi.coordinate));
|
|
286
|
+
}
|
|
287
|
+
catch (e) {
|
|
288
|
+
if (isDevMode())
|
|
289
|
+
console.warn('Error parsing POI coordinates', poi.coordinate);
|
|
290
|
+
}
|
|
291
|
+
}
|
|
292
|
+
if (poi.metadata) {
|
|
293
|
+
try {
|
|
294
|
+
const tagMetadata = JSON.parse(poi.metadata);
|
|
295
|
+
mattertagData.setNormal(tagMetadata.normal || { x: 0, y: -0.15, z: 0 });
|
|
296
|
+
}
|
|
297
|
+
catch (e) {
|
|
298
|
+
mattertagData.setNormal({ x: 0, y: -0.15, z: 0 });
|
|
299
|
+
}
|
|
300
|
+
}
|
|
301
|
+
if (poi.matterportSweepID) {
|
|
302
|
+
mattertagData.setSweepID(poi.matterportSweepID);
|
|
303
|
+
}
|
|
304
|
+
mattertagData.setPoi(poi);
|
|
305
|
+
const createdTagID = await this.addMattertagToViewer(sdk, mattertagData);
|
|
306
|
+
if (createdTagID && sdk) {
|
|
307
|
+
await wait(100);
|
|
308
|
+
await setTagIconAndOpacity(createdTagID, mattertagData);
|
|
309
|
+
injectHtmlInTag(tagType, object, createdTagID).catch(err => {
|
|
310
|
+
if (isDevMode())
|
|
311
|
+
console.warn('HTML injection failed for', createdTagID, err);
|
|
312
|
+
});
|
|
313
|
+
}
|
|
314
|
+
return createdTagID;
|
|
315
|
+
}
|
|
316
|
+
/** Updates the icon, opacity, and HTML content of an existing tag. */
|
|
317
|
+
async updateMatterTagContentForTagID(sdk, mattertagID, object, poiType, setTagIconAndOpacity, injectHtmlInTag) {
|
|
318
|
+
const mattertagData = this.dictionnaryTags.get(mattertagID);
|
|
319
|
+
if (!mattertagData)
|
|
320
|
+
return;
|
|
321
|
+
if (object) {
|
|
322
|
+
mattertagData.setObject(object, poiType);
|
|
323
|
+
}
|
|
324
|
+
await setTagIconAndOpacity(mattertagID, mattertagData);
|
|
325
|
+
await injectHtmlInTag(poiType, object, mattertagID);
|
|
326
|
+
}
|
|
327
|
+
/** Registers a custom texture and applies it to a tag, optionally adjusting opacity. */
|
|
328
|
+
async addNewIconAndSetForTag(sdk, mattertagID, iconPath, applyOpacity, opacity) {
|
|
329
|
+
if (!sdk)
|
|
330
|
+
return;
|
|
331
|
+
try {
|
|
332
|
+
const iconName = `custom-icon-${mattertagID}`;
|
|
333
|
+
await sdk.Asset.registerTexture(iconName, iconPath);
|
|
334
|
+
await sdk.Tag.editIcon(mattertagID, iconName);
|
|
335
|
+
if (applyOpacity) {
|
|
336
|
+
await sdk.Tag.editOpacity(mattertagID, opacity);
|
|
337
|
+
}
|
|
338
|
+
}
|
|
339
|
+
catch (error) {
|
|
340
|
+
if (isDevMode())
|
|
341
|
+
console.warn('Error setting custom icon or opacity', error);
|
|
342
|
+
}
|
|
343
|
+
}
|
|
344
|
+
/** Finds the tag ID and sweep ID associated with a specific element ID. */
|
|
345
|
+
getTagFromElementId(elementID, sweeps) {
|
|
346
|
+
let tagID = null;
|
|
347
|
+
let sweepID = null;
|
|
348
|
+
for (let [mattertagID, mattertagData] of this.dictionnaryTags) {
|
|
349
|
+
if (mattertagData.elementID === elementID) {
|
|
350
|
+
tagID = mattertagID;
|
|
351
|
+
const sweep = mattertagData.getSweepID();
|
|
352
|
+
if (sweep && sweeps && sweeps.includes(sweep)) {
|
|
353
|
+
sweepID = sweep;
|
|
354
|
+
}
|
|
355
|
+
}
|
|
356
|
+
}
|
|
357
|
+
return { tag: tagID, sweep: sweepID };
|
|
358
|
+
}
|
|
359
|
+
/** Retrieves the floor identifier associated with a tag based on its sweep data.
|
|
360
|
+
* Handles both the object form ({ floorInfo: { id, sequence }, floor: { id } })
|
|
361
|
+
* and the primitive form (floor: number) used by Matterport SDK v3.
|
|
362
|
+
*/
|
|
363
|
+
getFloorIdForTag(mattertagData, sweepCollection) {
|
|
364
|
+
const sweepId = mattertagData.getSweepID();
|
|
365
|
+
if (!sweepId || !sweepCollection)
|
|
366
|
+
return null;
|
|
367
|
+
const sweepData = sweepCollection[sweepId];
|
|
368
|
+
if (!sweepData)
|
|
369
|
+
return null;
|
|
370
|
+
// SDK v3: sweep.floor is a primitive number (the sequence index)
|
|
371
|
+
if (typeof sweepData.floor === 'number') {
|
|
372
|
+
return sweepData.floor.toString();
|
|
373
|
+
}
|
|
374
|
+
// Older / extended formats
|
|
375
|
+
return sweepData?.floorInfo?.id
|
|
376
|
+
?? sweepData?.floorInfo?.sequence?.toString()
|
|
377
|
+
?? sweepData?.floorId
|
|
378
|
+
?? sweepData?.floor?.id
|
|
379
|
+
?? sweepData?.floor?.sequence?.toString()
|
|
380
|
+
?? null;
|
|
381
|
+
}
|
|
382
|
+
/**
|
|
383
|
+
* Injects custom HTML or fallback billboard media into a Mattertag sandbox.
|
|
384
|
+
* Attaches event listeners for interactive elements within the tag.
|
|
385
|
+
*/
|
|
386
|
+
async injectHtmlInTag(sdk, tagType, object, tagID) {
|
|
387
|
+
if (!this.tagService) {
|
|
388
|
+
console.error('[MatterportTagService] tagService is not initialized.');
|
|
389
|
+
return;
|
|
390
|
+
}
|
|
391
|
+
try {
|
|
392
|
+
let html = await this.tagService.getHtmlToInject(tagType, object);
|
|
393
|
+
if (html && html.trim().length > 0 && sdk) {
|
|
394
|
+
const scriptTag = this.tagService.getScriptForTag(object, tagType);
|
|
395
|
+
html += `${scriptTag}`;
|
|
396
|
+
const [sandboxId, messenger] = await sdk.Tag.registerSandbox(html);
|
|
397
|
+
const attachmentID = this.tagsAttachments[tagID];
|
|
398
|
+
if (attachmentID) {
|
|
399
|
+
try {
|
|
400
|
+
await sdk.Tag.detach(tagID, attachmentID);
|
|
401
|
+
}
|
|
402
|
+
catch (e) { }
|
|
403
|
+
}
|
|
404
|
+
this.tagsAttachments[tagID] = sandboxId;
|
|
405
|
+
await sdk.Tag.attach(tagID, sandboxId);
|
|
406
|
+
// Attach the "View informations" handler in tooltip
|
|
407
|
+
messenger.on(TagAction.DETAIL_CLICK, this.tagService.onActionDetailClick.bind(this.tagService));
|
|
408
|
+
messenger.on(TagAction.TICKET_CLICK, this.tagService.onActionDetailClick.bind(this.tagService));
|
|
409
|
+
messenger.on(TagAction.AUDIO_CLICK, (audioCommentID) => this.tagService.onActionAudioClick(audioCommentID));
|
|
410
|
+
messenger.on(TagAction.VIDEO_CLICK, (videoUrl) => this.tagService.onActionVideoClick(videoUrl));
|
|
411
|
+
messenger.on(TagAction.IMAGE_CLICK, (imageCommentID) => this.tagService.onActionImageClick(imageCommentID));
|
|
412
|
+
messenger.on(TagAction.DOC_CLICK, (docUrl) => this.tagService.onActionDocClick(docUrl));
|
|
413
|
+
messenger.on(TagAction.YOUTUBE_CLICK, (youtubeUrl) => this.tagService.onActionYoutubeClick(youtubeUrl));
|
|
414
|
+
}
|
|
415
|
+
else if (sdk) {
|
|
416
|
+
const { comment, tagDescription } = this.tagService.getBillboardMediaToEmbed(object);
|
|
417
|
+
if (comment) {
|
|
418
|
+
const attachmentID = await sdk.Tag.registerAttachment({
|
|
419
|
+
type: 'media',
|
|
420
|
+
data: {
|
|
421
|
+
url: comment.externalLink,
|
|
422
|
+
label: object.title || 'Media',
|
|
423
|
+
},
|
|
424
|
+
});
|
|
425
|
+
await sdk.Tag.attach(tagID, attachmentID);
|
|
426
|
+
const billboardAttachmentID = await sdk.Tag.registerAttachment({
|
|
427
|
+
type: 'billboard',
|
|
428
|
+
data: {
|
|
429
|
+
title: object.title,
|
|
430
|
+
description: tagDescription,
|
|
431
|
+
},
|
|
432
|
+
});
|
|
433
|
+
await sdk.Tag.attach(tagID, billboardAttachmentID);
|
|
434
|
+
this.tagsAttachments[tagID] = billboardAttachmentID;
|
|
435
|
+
}
|
|
436
|
+
}
|
|
437
|
+
}
|
|
438
|
+
catch (error) {
|
|
439
|
+
console.error('[MatterportTagService] Error in injectHtmlInTag:', error);
|
|
440
|
+
}
|
|
441
|
+
}
|
|
442
|
+
/** Clears all local tag data and resets the addition queue. */
|
|
443
|
+
clear() {
|
|
444
|
+
this.dictionnaryTags.clear();
|
|
445
|
+
this.tagAddQueue = Promise.resolve();
|
|
446
|
+
this.mattertagToFollow = null;
|
|
447
|
+
this.mattertagIDs = [];
|
|
448
|
+
}
|
|
449
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: MatterportTagService, deps: [{ token: i1.MatterportNavigationService }], target: i0.ɵɵFactoryTarget.Injectable });
|
|
450
|
+
static ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: MatterportTagService, providedIn: 'root' });
|
|
451
|
+
}
|
|
452
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: MatterportTagService, decorators: [{
|
|
453
|
+
type: Injectable,
|
|
454
|
+
args: [{
|
|
455
|
+
providedIn: 'root',
|
|
456
|
+
}]
|
|
457
|
+
}], ctorParameters: () => [{ type: i1.MatterportNavigationService }] });
|
|
458
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"matterport-tag.service.js","sourceRoot":"","sources":["../../../../../projects/ngx-smarterplan-core/src/lib/services/matterport-tag.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { isDevMode, Injectable } from '@angular/core';\r\nimport { Subject } from 'rxjs';\r\nimport { MattertagData } from '../mattertagData';\r\nimport { PoiType, DbObjectType, POI, ViewerInteractions, TagAction, CameraMode } from '../types.service';\r\nimport { MatterportNavigationService } from './matterport-navigation.service';\r\nimport { wait } from '../helpers.service';\r\nimport { BaseTagService } from './tag.service';\r\n\r\n@Injectable({\r\n  providedIn: 'root',\r\n})\r\nexport class MatterportTagService {\r\n  public dictionnaryTags: Map<string, MattertagData> = new Map();\r\n  public tagAddQueue: Promise<any> = Promise.resolve();\r\n  public mattertagToFollow: string | null = null;\r\n  public onGoToTag = new Subject<string>();\r\n  public mattertagIDs: Array<string> = [];\r\n  public tagsAttachments: Object = {};\r\n  public tagMessengerOn: boolean = false;\r\n  public tagService: BaseTagService | null = null;\r\n  private tagsCurrentOpacity: Map<string, number> = new Map();\r\n  private tagsCurrentEnabled: Map<string, boolean> = new Map();\r\n\r\n  constructor(\r\n    private navigationService: MatterportNavigationService\r\n  ) { }\r\n\r\n  setTagService(service: BaseTagService): void {\r\n    this.tagService = service;\r\n  }\r\n\r\n  /** Opens a specific tag in the Matterport viewer by its ID. */\r\n  async goToTag(sdk: any, sid: string): Promise<any> {\r\n    if (!sdk) return;\r\n    try {\r\n      this.onGoToTag.next(sid);\r\n      await sdk.Tag.open(sid);\r\n    } catch (e) {\r\n      if (isDevMode()) console.warn('Cannot open tag', sid, e);\r\n    }\r\n  }\r\n\r\n  /** Opens the most recently created tag in the viewer. */\r\n  async goToLastTag(sdk: any): Promise<any> {\r\n    if (!sdk || this.mattertagIDs.length === 0) return;\r\n    const lastSid = this.mattertagIDs[this.mattertagIDs.length - 1];\r\n    return this.goToTag(sdk, lastSid);\r\n  }\r\n\r\n  /**\r\n   * Adds a Mattertag to the viewer, serializing calls to prevent SDK race conditions.\r\n   * Reuses existing IDs if the tag is already present in the session.\r\n   */\r\n  async addMattertagToViewer(sdk: any, mattertagData: MattertagData): Promise<string | null> {\r\n    if (!sdk) return null;\r\n\r\n    this.tagAddQueue = this.tagAddQueue.then(async () => {\r\n      const data = mattertagData.getData() as any;\r\n      const existingId = data?.id || null;\r\n\r\n      try {\r\n        const sidList = await sdk.Tag.add(data);\r\n        if (sidList && sidList.length > 0) {\r\n          const mattertagID = sidList[0];\r\n          this.mattertagIDs.push(mattertagID);\r\n          this.dictionnaryTags.set(mattertagID, mattertagData);\r\n          return mattertagID;\r\n        }\r\n      } catch (error: any) {\r\n        if (error?.message?.includes('already in use') && existingId) {\r\n          if (!this.dictionnaryTags.has(existingId)) {\r\n            this.mattertagIDs.push(existingId);\r\n            this.dictionnaryTags.set(existingId, mattertagData);\r\n          }\r\n          return existingId;\r\n        }\r\n        if (isDevMode()) console.warn('Failed to add tag to viewer', error);\r\n      }\r\n      return null;\r\n    });\r\n\r\n    return this.tagAddQueue;\r\n  }\r\n\r\n  /**\r\n   * Internal method to add and configure a Mattertag with icon, opacity, and HTML content.\r\n   * Handles existing tags gracefully and queues operations to avoid SDK conflicts.\r\n   */\r\n  async _doAddMattertag(\r\n    sdk: any,\r\n    mattertagData: MattertagData,\r\n    setTagIconAndOpacity: (sid: string, mt: MattertagData) => Promise<void>,\r\n    injectHtmlInTag: (type: PoiType, obj: DbObjectType, sid: string) => Promise<void>\r\n  ): Promise<string> {\r\n    const descriptor = mattertagData.getMattertagDescriptor();\r\n    //console.log('descriptor', descriptor);\r\n    const sid = descriptor[0]['id'];\r\n\r\n    if (sid && this.dictionnaryTags.has(sid)) {\r\n      await setTagIconAndOpacity(sid, mattertagData);\r\n      await injectHtmlInTag(mattertagData.getType(), mattertagData.getObject(), sid);\r\n      return sid;\r\n    }\r\n\r\n    this.tagAddQueue = this.tagAddQueue.then(async () => {\r\n      try {\r\n        const [createdSid] = await sdk.Tag.add(...descriptor);\r\n        this.mattertagIDs.push(createdSid);\r\n        this.dictionnaryTags.set(createdSid, mattertagData);\r\n\r\n        await wait(100);\r\n        await setTagIconAndOpacity(createdSid, mattertagData);\r\n\r\n        injectHtmlInTag(mattertagData.getType(), mattertagData.getObject(), createdSid).catch(err => {\r\n          if (isDevMode()) console.warn('HTML injection failed for', createdSid, err);\r\n        });\r\n\r\n        return createdSid;\r\n      } catch (e: any) {\r\n        if (e?.message?.includes('already in use') && sid) {\r\n          if (!this.dictionnaryTags.has(sid)) {\r\n            this.mattertagIDs.push(sid);\r\n            this.dictionnaryTags.set(sid, mattertagData);\r\n          }\r\n          await setTagIconAndOpacity(sid, mattertagData);\r\n          await injectHtmlInTag(mattertagData.getType(), mattertagData.getObject(), sid);\r\n          return sid;\r\n        }\r\n        throw e;\r\n      }\r\n    });\r\n\r\n    return this.tagAddQueue;\r\n  }\r\n\r\n  getDistance(p1: any, p2: any): number {\r\n    return Math.sqrt(Math.pow(p2.x - p1.x, 2) + Math.pow(p2.y - p1.y, 2) + Math.pow(p2.z - p1.z, 2));\r\n  }\r\n\r\n  /**\r\n   * Checks if a tag is within the bounding box of any active room (adding a slight margin to account for wall-mounted tags).\r\n   * If there are no active rooms (e.g., hallway), falls back to returning true.\r\n   */\r\n  isTagInRooms(mattertagData: MattertagData, rooms: any[]): boolean {\r\n    if (!rooms || rooms.length === 0) return false;\r\n\r\n    const tagPos = mattertagData.getPosition();\r\n    const margin = 1.5;\r\n\r\n    return rooms.some((room) => {\r\n      if (!room.bounds) return false;\r\n      const { min, max } = room.bounds;\r\n\r\n      return (\r\n        tagPos.x >= min.x - margin && tagPos.x <= max.x + margin &&\r\n        tagPos.y >= min.y - margin && tagPos.y <= max.y + margin &&\r\n        tagPos.z >= min.z - margin && tagPos.z <= max.z + margin\r\n      );\r\n    });\r\n  }\r\n\r\n  /**\r\n   * Computes the final context-aware opacity for a single tag.\r\n   * @param currentFloorSequence  Authoritative current floor sequence from sdk.Floor.current\r\n   *                              (null = single-floor model or unknown).\r\n   * @param currentRooms          Array of active rooms the camera is in.\r\n   */\r\n  /*computeSingleTagContextOpacity(\r\n    mattertagData: MattertagData,\r\n    poseCamera: any,\r\n    sweepCollection: any,\r\n    currentFloorSequence: number | null,\r\n    currentRooms: any[]\r\n  ): number {\r\n    const mode = this.navigationService.currentCameraMode;\r\n\r\n    // 1. Check Floor\r\n    const tagFloorId = this.getFloorIdForTag(mattertagData, sweepCollection);\r\n    const currentFloorId = currentFloorSequence?.toString();\r\n    if (tagFloorId && currentFloorId && tagFloorId !== currentFloorId) {\r\n      return 0;\r\n    }\r\n\r\n    // 2. Si Dollhouse -> Pleine opacité\r\n    //if (mode !== CameraMode.INSIDE) return 1;\r\n\r\n    // 3. Si Inside -> Check pièce + Distance progressive\r\n    //if (!this.isTagInRooms(mattertagData, currentRooms)) return 0;\r\n\r\n    return 1;\r\n  }*/\r\n\r\n  /** Returns the ID of the most recently added tag. */\r\n  getLastTag(): string | null {\r\n    if (this.mattertagIDs.length === 0) return null;\r\n    return this.mattertagIDs[this.mattertagIDs.length - 1];\r\n  }\r\n\r\n  /**\r\n   * Updates the visibility (opacity) of all tags based on the current camera mode,\r\n   * floor and room context. Must be called after each sweep, floor or mode change.\r\n   *\r\n   * Rules:\r\n   *  - DOLLHOUSE / FLOORPLAN → all tags hidden (opacity 0).\r\n   *  - INSIDE → only tags whose sweep belongs to the current floor are shown;\r\n   *    proximity-based opacity is applied via computeTagOpacity().\r\n   *    Tags on other floors are hidden (opacity 0).\r\n   *  - TRANSITIONING → no change (skip update to avoid flickering).\r\n   *\r\n   * @param sdk          Matterport SDK instance.\r\n   * @param poseCamera   Current camera pose.\r\n   * @param sweepCollection  Full collection of sweeps.\r\n   * @param currentCameraMode  Current camera mode.\r\n   * @param mattertagToFollow  ID of the cursor tag (never touched).\r\n   */\r\n  async updateTagsVisibilityForContext(\r\n    sdk: any,\r\n    sweepCollection: any,\r\n    currentCameraMode: CameraMode,\r\n    mattertagToFollow: string | null,\r\n    currentFloorSequence: number | null = null,\r\n    currentRooms: any[] = []\r\n  ): Promise<void> {\r\n    if (!sdk || this.dictionnaryTags.size === 0) return;\r\n\r\n    const isInteriorMode = currentCameraMode === CameraMode.INSIDE;\r\n    const currentFloorId = currentFloorSequence !== null ? currentFloorSequence.toString() : null;\r\n    const promises = [];\r\n\r\n    for (const [tagId, mattertagData] of this.dictionnaryTags) {\r\n      if (tagId === mattertagToFollow) continue;\r\n\r\n      const tagFloorId = this.getFloorIdForTag(mattertagData, sweepCollection);\r\n      const isOnCurrentFloor = !tagFloorId || !currentFloorId || tagFloorId === currentFloorId;\r\n\r\n      let targetOpacity = 0;\r\n      let shouldBeEnabled = false;\r\n\r\n      // Calculation of opacity and enable status\r\n      if (isOnCurrentFloor) {\r\n        if (!isInteriorMode) {\r\n          targetOpacity = 1;\r\n          shouldBeEnabled = true;\r\n        } else {\r\n          targetOpacity = 0.9;\r\n          shouldBeEnabled = true;\r\n        }\r\n      }\r\n      // Apply changes\r\n\r\n      // Tooltip and click management\r\n      const lastEnabled = this.tagsCurrentEnabled.get(tagId);\r\n      if (lastEnabled !== shouldBeEnabled) {\r\n        this.tagsCurrentEnabled.set(tagId, shouldBeEnabled);\r\n        promises.push(sdk.Tag.allowAction(tagId, {\r\n          opening: shouldBeEnabled,\r\n          navigating: shouldBeEnabled,\r\n          docking: false\r\n        }).catch(() => { }));\r\n      }\r\n      // Opacity management\r\n      const lastOpacity = this.tagsCurrentOpacity.get(tagId) ?? -1;\r\n      if (Math.abs(lastOpacity - targetOpacity) > 0.05) {\r\n        this.tagsCurrentOpacity.set(tagId, targetOpacity);\r\n        promises.push(sdk.Tag.editOpacity(tagId, targetOpacity).catch(() => { }));\r\n      }\r\n    }\r\n\r\n    await Promise.all(promises);\r\n  }\r\n\r\n  /** Removes a specific tag from the viewer and local dictionaries. */\r\n  async deleteMattertagFromId(sdk: any, mattertagID: string): Promise<void> {\r\n    try {\r\n      if (sdk) await sdk.Tag.remove(mattertagID);\r\n      this.dictionnaryTags.delete(mattertagID);\r\n      const index = this.mattertagIDs.indexOf(mattertagID);\r\n      if (index > -1) this.mattertagIDs.splice(index, 1);\r\n    } catch (e: any) {\r\n      if (isDevMode()) console.warn('Cannot delete tag', mattertagID, e);\r\n    }\r\n  }\r\n\r\n  /** Removes the last added tag from the viewer. */\r\n  async deleteLastMattertag(sdk: any): Promise<void> {\r\n    const lastSid = this.getLastTag();\r\n    if (lastSid) await this.deleteMattertagFromId(sdk, lastSid);\r\n  }\r\n\r\n  /** Clears all tags from the viewer and resets local state. */\r\n  async action_delete_all_mattertags(sdk: any): Promise<void> {\r\n    if (sdk) {\r\n      const sidsToRemove = [...this.mattertagIDs];\r\n      for (const sid of sidsToRemove) {\r\n        try {\r\n          await sdk.Tag.remove(sid);\r\n        } catch (e) { }\r\n      }\r\n    }\r\n    this.clear();\r\n  }\r\n\r\n  /**\r\n   * Creates and configures a Mattertag from a POI, handling coordinates, normals, and sweep ID.\r\n   * Skips creation if a tag for the object already exists.\r\n   */\r\n  async createMattertagFromPOI(\r\n    sdk: any,\r\n    tagType: PoiType,\r\n    object: DbObjectType,\r\n    poi: POI,\r\n    poseCamera: any,\r\n    getTagFromElementId: (id: string) => { tag: string; sweep: string },\r\n    setTagIconAndOpacity: (sid: string, mt: MattertagData) => Promise<void>,\r\n    injectHtmlInTag: (type: PoiType, obj: DbObjectType, sid: string) => Promise<void>\r\n  ): Promise<any> {\r\n    const { tag } = getTagFromElementId(object.id);\r\n    if (tag) return;\r\n\r\n    const mattertagData: MattertagData = new MattertagData(tagType);\r\n    mattertagData.setObject(object, tagType);\r\n\r\n    if (poi.coordinate) {\r\n      try {\r\n        mattertagData.setPosition(JSON.parse(poi.coordinate));\r\n      } catch (e) {\r\n        if (isDevMode()) console.warn('Error parsing POI coordinates', poi.coordinate);\r\n      }\r\n    }\r\n\r\n    if (poi.metadata) {\r\n      try {\r\n        const tagMetadata = JSON.parse(poi.metadata);\r\n        mattertagData.setNormal(tagMetadata.normal || { x: 0, y: -0.15, z: 0 });\r\n      } catch (e) {\r\n        mattertagData.setNormal({ x: 0, y: -0.15, z: 0 });\r\n      }\r\n    }\r\n\r\n    if (poi.matterportSweepID) {\r\n      mattertagData.setSweepID(poi.matterportSweepID);\r\n    }\r\n    mattertagData.setPoi(poi);\r\n\r\n    const createdTagID = await this.addMattertagToViewer(sdk, mattertagData);\r\n\r\n    if (createdTagID && sdk) {\r\n      await wait(100);\r\n      await setTagIconAndOpacity(createdTagID, mattertagData);\r\n      injectHtmlInTag(tagType, object, createdTagID).catch(err => {\r\n        if (isDevMode()) console.warn('HTML injection failed for', createdTagID, err);\r\n      });\r\n    }\r\n\r\n    return createdTagID;\r\n  }\r\n\r\n  /** Updates the icon, opacity, and HTML content of an existing tag. */\r\n  async updateMatterTagContentForTagID(\r\n    sdk: any,\r\n    mattertagID: string,\r\n    object: DbObjectType,\r\n    poiType: PoiType,\r\n    setTagIconAndOpacity: (sid: string, mt: MattertagData) => Promise<void>,\r\n    injectHtmlInTag: (type: PoiType, obj: DbObjectType, sid: string) => Promise<void>\r\n  ): Promise<any> {\r\n    const mattertagData = this.dictionnaryTags.get(mattertagID);\r\n    if (!mattertagData) return;\r\n\r\n    if (object) {\r\n      mattertagData.setObject(object, poiType);\r\n    }\r\n\r\n    await setTagIconAndOpacity(mattertagID, mattertagData);\r\n    await injectHtmlInTag(poiType, object, mattertagID);\r\n  }\r\n\r\n  /** Registers a custom texture and applies it to a tag, optionally adjusting opacity. */\r\n  async addNewIconAndSetForTag(\r\n    sdk: any,\r\n    mattertagID: string,\r\n    iconPath: string,\r\n    applyOpacity: boolean,\r\n    opacity: number\r\n  ) {\r\n    if (!sdk) return;\r\n    try {\r\n      const iconName = `custom-icon-${mattertagID}`;\r\n      await sdk.Asset.registerTexture(iconName, iconPath);\r\n      await sdk.Tag.editIcon(mattertagID, iconName);\r\n\r\n      if (applyOpacity) {\r\n        await sdk.Tag.editOpacity(mattertagID, opacity);\r\n      }\r\n    } catch (error) {\r\n      if (isDevMode()) console.warn('Error setting custom icon or opacity', error);\r\n    }\r\n  }\r\n\r\n  /** Finds the tag ID and sweep ID associated with a specific element ID. */\r\n  getTagFromElementId(elementID: string, sweeps: string[] | null): { tag: string | null; sweep: string | null } {\r\n    let tagID = null;\r\n    let sweepID = null;\r\n    for (let [mattertagID, mattertagData] of this.dictionnaryTags) {\r\n      if (mattertagData.elementID === elementID) {\r\n        tagID = mattertagID;\r\n        const sweep = mattertagData.getSweepID();\r\n        if (sweep && sweeps && sweeps.includes(sweep)) {\r\n          sweepID = sweep;\r\n        }\r\n      }\r\n    }\r\n    return { tag: tagID, sweep: sweepID };\r\n  }\r\n\r\n  /** Retrieves the floor identifier associated with a tag based on its sweep data.\r\n   *  Handles both the object form ({ floorInfo: { id, sequence }, floor: { id } })\r\n   *  and the primitive form (floor: number) used by Matterport SDK v3.\r\n   */\r\n  getFloorIdForTag(mattertagData: MattertagData, sweepCollection: any): string | null {\r\n    const sweepId = mattertagData.getSweepID();\r\n    if (!sweepId || !sweepCollection) return null;\r\n\r\n    const sweepData = sweepCollection[sweepId];\r\n    if (!sweepData) return null;\r\n\r\n    // SDK v3: sweep.floor is a primitive number (the sequence index)\r\n    if (typeof sweepData.floor === 'number') {\r\n      return sweepData.floor.toString();\r\n    }\r\n\r\n    // Older / extended formats\r\n    return sweepData?.floorInfo?.id\r\n      ?? sweepData?.floorInfo?.sequence?.toString()\r\n      ?? sweepData?.floorId\r\n      ?? sweepData?.floor?.id\r\n      ?? sweepData?.floor?.sequence?.toString()\r\n      ?? null;\r\n  }\r\n\r\n  /**\r\n   * Injects custom HTML or fallback billboard media into a Mattertag sandbox.\r\n   * Attaches event listeners for interactive elements within the tag.\r\n   */\r\n  async injectHtmlInTag(sdk: any, tagType: PoiType, object: DbObjectType, tagID: string) {\r\n    if (!this.tagService) {\r\n      console.error('[MatterportTagService] tagService is not initialized.');\r\n      return;\r\n    }\r\n    try {\r\n      let html = await this.tagService.getHtmlToInject(tagType, object);\r\n\r\n      if (html && html.trim().length > 0 && sdk) {\r\n        const scriptTag = this.tagService.getScriptForTag(object, tagType);\r\n        html += `${scriptTag}`;\r\n\r\n        const [sandboxId, messenger] = await sdk.Tag.registerSandbox(html);\r\n\r\n        const attachmentID = this.tagsAttachments[tagID];\r\n        if (attachmentID) {\r\n          try {\r\n            await sdk.Tag.detach(tagID, attachmentID);\r\n          } catch (e) { }\r\n        }\r\n\r\n        this.tagsAttachments[tagID] = sandboxId;\r\n        await sdk.Tag.attach(tagID, sandboxId);\r\n\r\n        // Attach the \"View informations\" handler in tooltip\r\n        messenger.on(TagAction.DETAIL_CLICK, this.tagService.onActionDetailClick.bind(this.tagService));\r\n        messenger.on(TagAction.TICKET_CLICK, this.tagService.onActionDetailClick.bind(this.tagService));\r\n        messenger.on(TagAction.AUDIO_CLICK, (audioCommentID: string) => this.tagService.onActionAudioClick(audioCommentID));\r\n        messenger.on(TagAction.VIDEO_CLICK, (videoUrl: string) => this.tagService.onActionVideoClick(videoUrl));\r\n        messenger.on(TagAction.IMAGE_CLICK, (imageCommentID: string) => this.tagService.onActionImageClick(imageCommentID));\r\n        messenger.on(TagAction.DOC_CLICK, (docUrl: string) => this.tagService.onActionDocClick(docUrl));\r\n        messenger.on(TagAction.YOUTUBE_CLICK, (youtubeUrl: string) => this.tagService.onActionYoutubeClick(youtubeUrl));\r\n      } else if (sdk) {\r\n        const { comment, tagDescription } = this.tagService.getBillboardMediaToEmbed(object);\r\n\r\n        if (comment) {\r\n          const attachmentID = await sdk.Tag.registerAttachment({\r\n            type: 'media',\r\n            data: {\r\n              url: comment.externalLink,\r\n              label: object.title || 'Media',\r\n            },\r\n          });\r\n\r\n          await sdk.Tag.attach(tagID, attachmentID);\r\n\r\n          const billboardAttachmentID = await sdk.Tag.registerAttachment({\r\n            type: 'billboard',\r\n            data: {\r\n              title: object.title,\r\n              description: tagDescription,\r\n            },\r\n          });\r\n\r\n          await sdk.Tag.attach(tagID, billboardAttachmentID);\r\n          this.tagsAttachments[tagID] = billboardAttachmentID;\r\n        }\r\n      }\r\n    } catch (error) {\r\n      console.error('[MatterportTagService] Error in injectHtmlInTag:', error);\r\n    }\r\n  }\r\n\r\n  /** Clears all local tag data and resets the addition queue. */\r\n  clear() {\r\n    this.dictionnaryTags.clear();\r\n    this.tagAddQueue = Promise.resolve();\r\n    this.mattertagToFollow = null;\r\n    this.mattertagIDs = [];\r\n  }\r\n}\r\n"]}
|