@inweb/viewer-three 26.10.6 → 26.12.0

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.
Files changed (153) hide show
  1. package/README.md +7 -4
  2. package/dist/{plugins → extensions}/components/AxesHelperComponent.js +23 -1
  3. package/dist/extensions/components/AxesHelperComponent.js.map +1 -0
  4. package/dist/extensions/components/AxesHelperComponent.min.js +24 -0
  5. package/dist/{plugins → extensions}/components/AxesHelperComponent.module.js +24 -2
  6. package/dist/extensions/components/AxesHelperComponent.module.js.map +1 -0
  7. package/dist/{plugins → extensions}/components/ExtentsHelperComponent.js +18 -0
  8. package/dist/extensions/components/ExtentsHelperComponent.js.map +1 -0
  9. package/dist/{plugins/components/AxesHelperComponent.min.js → extensions/components/ExtentsHelperComponent.min.js} +1 -1
  10. package/dist/{plugins → extensions}/components/ExtentsHelperComponent.module.js +19 -1
  11. package/dist/extensions/components/ExtentsHelperComponent.module.js.map +1 -0
  12. package/dist/extensions/components/GridHelperComponent.js.map +1 -0
  13. package/dist/extensions/components/GridHelperComponent.module.js.map +1 -0
  14. package/dist/extensions/components/InfoPanelComponent.js +170 -0
  15. package/dist/extensions/components/InfoPanelComponent.js.map +1 -0
  16. package/dist/extensions/components/InfoPanelComponent.min.js +24 -0
  17. package/dist/extensions/components/InfoPanelComponent.module.js +164 -0
  18. package/dist/extensions/components/InfoPanelComponent.module.js.map +1 -0
  19. package/dist/extensions/components/LightHelperComponent.js.map +1 -0
  20. package/dist/extensions/components/LightHelperComponent.module.js.map +1 -0
  21. package/dist/extensions/components/RoomEnvironmentComponent.js.map +1 -0
  22. package/dist/extensions/components/RoomEnvironmentComponent.module.js.map +1 -0
  23. package/dist/{plugins → extensions}/components/StatsPanelComponent.js +9 -3
  24. package/dist/extensions/components/StatsPanelComponent.js.map +1 -0
  25. package/dist/extensions/components/StatsPanelComponent.min.js +24 -0
  26. package/dist/{plugins → extensions}/components/StatsPanelComponent.module.js +9 -3
  27. package/dist/extensions/components/StatsPanelComponent.module.js.map +1 -0
  28. package/dist/{plugins → extensions}/loaders/GLTFCloudLoader.js +2 -3
  29. package/dist/extensions/loaders/GLTFCloudLoader.js.map +1 -0
  30. package/dist/{plugins → extensions}/loaders/GLTFCloudLoader.min.js +1 -1
  31. package/dist/{plugins → extensions}/loaders/GLTFCloudLoader.module.js +2 -3
  32. package/dist/extensions/loaders/GLTFCloudLoader.module.js.map +1 -0
  33. package/dist/extensions/loaders/GLTFFileLoader.js +2499 -0
  34. package/dist/extensions/loaders/GLTFFileLoader.js.map +1 -0
  35. package/dist/extensions/loaders/GLTFFileLoader.min.js +24 -0
  36. package/dist/extensions/loaders/GLTFFileLoader.module.js +74 -0
  37. package/dist/extensions/loaders/GLTFFileLoader.module.js.map +1 -0
  38. package/dist/{plugins → extensions}/loaders/IFCXLoader.js +5 -7
  39. package/dist/extensions/loaders/IFCXLoader.js.map +1 -0
  40. package/dist/{plugins → extensions}/loaders/IFCXLoader.min.js +1 -1
  41. package/dist/{plugins → extensions}/loaders/IFCXLoader.module.js +5 -7
  42. package/dist/extensions/loaders/IFCXLoader.module.js.map +1 -0
  43. package/dist/{plugins → extensions}/loaders/PotreeLoader.js +56 -6
  44. package/dist/extensions/loaders/PotreeLoader.js.map +1 -0
  45. package/dist/extensions/loaders/PotreeLoader.min.js +24 -0
  46. package/dist/{plugins → extensions}/loaders/PotreeLoader.module.js +53 -2
  47. package/dist/extensions/loaders/PotreeLoader.module.js.map +1 -0
  48. package/dist/viewer-three.js +1416 -2930
  49. package/dist/viewer-three.js.map +1 -1
  50. package/dist/viewer-three.min.js +8 -3
  51. package/dist/viewer-three.module.js +1205 -363
  52. package/dist/viewer-three.module.js.map +1 -1
  53. package/{plugins → extensions}/components/AxesHelperComponent.ts +31 -2
  54. package/{plugins → extensions}/components/ExtentsHelperComponent.ts +25 -0
  55. package/extensions/components/InfoPanelComponent.ts +197 -0
  56. package/{plugins → extensions}/components/StatsPanelComponent.ts +10 -3
  57. package/{plugins → extensions}/loaders/GLTFCloudLoader.ts +2 -3
  58. package/{src/Viewer → extensions}/loaders/GLTFFileLoader.ts +21 -12
  59. package/{plugins → extensions}/loaders/IFCX/IFCXCloudLoader.ts +5 -5
  60. package/{plugins → extensions}/loaders/IFCX/IFCXFileLoader.ts +3 -4
  61. package/{plugins → extensions}/loaders/Potree/PotreeFileLoader.ts +3 -4
  62. package/extensions/loaders/Potree/PotreeModelImpl.ts +108 -0
  63. package/lib/Viewer/Viewer.d.ts +28 -20
  64. package/lib/Viewer/commands/GetSelected2.d.ts +2 -0
  65. package/lib/Viewer/commands/SelectModel.d.ts +1 -1
  66. package/lib/Viewer/commands/SetSelected2.d.ts +2 -0
  67. package/lib/Viewer/components/InfoComponent.d.ts +22 -0
  68. package/lib/Viewer/components/SelectionComponent.d.ts +1 -3
  69. package/lib/Viewer/components/index.d.ts +6 -6
  70. package/lib/Viewer/draggers/MeasureLineDragger.d.ts +7 -1
  71. package/lib/Viewer/loaders/DynamicGltfLoader/DynamicModelImpl.d.ts +2 -1
  72. package/lib/Viewer/loaders/GLTFBinaryExtension.d.ts +5 -0
  73. package/lib/Viewer/loaders/GLTFCloudDynamicLoader.d.ts +2 -2
  74. package/lib/Viewer/loaders/{GLTFFileLoader.d.ts → GLTFFileDynamicLoader.d.ts} +7 -1
  75. package/lib/Viewer/loaders/GLTFLoadingManager.d.ts +4 -3
  76. package/lib/Viewer/loaders/RangesLoader.d.ts +15 -0
  77. package/lib/Viewer/loaders/index.d.ts +22 -14
  78. package/lib/Viewer/measurement/Snapper.d.ts +15 -0
  79. package/lib/Viewer/measurement/UnitConverter.d.ts +63 -0
  80. package/lib/Viewer/measurement/UnitFormatter.d.ts +4 -0
  81. package/lib/Viewer/models/IModelImpl.d.ts +11 -8
  82. package/lib/Viewer/models/ModelImpl.d.ts +9 -5
  83. package/package.json +11 -11
  84. package/src/Viewer/Viewer.ts +127 -88
  85. package/src/Viewer/commands/ClearSelected.ts +3 -1
  86. package/src/Viewer/commands/GetModels.ts +1 -1
  87. package/src/Viewer/commands/GetSelected.ts +2 -2
  88. package/{plugins/loaders/Potree/PotreeModelImpl.ts → src/Viewer/commands/GetSelected2.ts} +7 -9
  89. package/src/Viewer/commands/HideSelected.ts +3 -1
  90. package/src/Viewer/commands/SelectModel.ts +5 -5
  91. package/src/Viewer/commands/SetSelected.ts +9 -10
  92. package/src/Viewer/commands/SetSelected2.ts +42 -0
  93. package/src/Viewer/commands/ZoomToObjects.ts +5 -6
  94. package/src/Viewer/commands/ZoomToSelected.ts +3 -1
  95. package/src/Viewer/commands/index.ts +4 -0
  96. package/src/Viewer/components/CameraComponent.ts +6 -1
  97. package/src/Viewer/components/ExtentsComponent.ts +4 -1
  98. package/src/Viewer/components/InfoComponent.ts +187 -0
  99. package/src/Viewer/components/SelectionComponent.ts +7 -30
  100. package/src/Viewer/components/index.ts +8 -6
  101. package/src/Viewer/draggers/MeasureLineDragger.ts +84 -226
  102. package/src/Viewer/loaders/DynamicGltfLoader/DynamicGltfLoader.js +276 -39
  103. package/src/Viewer/loaders/DynamicGltfLoader/DynamicModelImpl.ts +45 -10
  104. package/src/Viewer/loaders/DynamicGltfLoader/GltfStructure.js +71 -2
  105. package/src/Viewer/loaders/GLTFBinaryExtension.ts +91 -0
  106. package/src/Viewer/loaders/GLTFCloudDynamicLoader.ts +13 -19
  107. package/src/Viewer/loaders/GLTFFileDynamicLoader.ts +145 -0
  108. package/src/Viewer/loaders/GLTFLoadingManager.ts +5 -4
  109. package/src/Viewer/loaders/RangesLoader.ts +105 -0
  110. package/src/Viewer/loaders/index.ts +24 -16
  111. package/src/Viewer/measurement/Snapper.ts +208 -0
  112. package/src/Viewer/measurement/UnitConverter.ts +47 -0
  113. package/src/Viewer/measurement/UnitFormatter.ts +95 -0
  114. package/src/Viewer/models/IModelImpl.ts +17 -8
  115. package/src/Viewer/models/ModelImpl.ts +205 -16
  116. package/src/index-umd.ts +1 -1
  117. package/dist/plugins/components/AxesHelperComponent.js.map +0 -1
  118. package/dist/plugins/components/AxesHelperComponent.module.js.map +0 -1
  119. package/dist/plugins/components/ExtentsHelperComponent.js.map +0 -1
  120. package/dist/plugins/components/ExtentsHelperComponent.min.js +0 -24
  121. package/dist/plugins/components/ExtentsHelperComponent.module.js.map +0 -1
  122. package/dist/plugins/components/GridHelperComponent.js.map +0 -1
  123. package/dist/plugins/components/GridHelperComponent.module.js.map +0 -1
  124. package/dist/plugins/components/LightHelperComponent.js.map +0 -1
  125. package/dist/plugins/components/LightHelperComponent.module.js.map +0 -1
  126. package/dist/plugins/components/RoomEnvironmentComponent.js.map +0 -1
  127. package/dist/plugins/components/RoomEnvironmentComponent.module.js.map +0 -1
  128. package/dist/plugins/components/StatsPanelComponent.js.map +0 -1
  129. package/dist/plugins/components/StatsPanelComponent.min.js +0 -24
  130. package/dist/plugins/components/StatsPanelComponent.module.js.map +0 -1
  131. package/dist/plugins/loaders/GLTFCloudLoader.js.map +0 -1
  132. package/dist/plugins/loaders/GLTFCloudLoader.module.js.map +0 -1
  133. package/dist/plugins/loaders/IFCXLoader.js.map +0 -1
  134. package/dist/plugins/loaders/IFCXLoader.module.js.map +0 -1
  135. package/dist/plugins/loaders/PotreeLoader.js.map +0 -1
  136. package/dist/plugins/loaders/PotreeLoader.min.js +0 -24
  137. package/dist/plugins/loaders/PotreeLoader.module.js.map +0 -1
  138. /package/dist/{plugins → extensions}/components/GridHelperComponent.js +0 -0
  139. /package/dist/{plugins → extensions}/components/GridHelperComponent.min.js +0 -0
  140. /package/dist/{plugins → extensions}/components/GridHelperComponent.module.js +0 -0
  141. /package/dist/{plugins → extensions}/components/LightHelperComponent.js +0 -0
  142. /package/dist/{plugins → extensions}/components/LightHelperComponent.min.js +0 -0
  143. /package/dist/{plugins → extensions}/components/LightHelperComponent.module.js +0 -0
  144. /package/dist/{plugins → extensions}/components/RoomEnvironmentComponent.js +0 -0
  145. /package/dist/{plugins → extensions}/components/RoomEnvironmentComponent.min.js +0 -0
  146. /package/dist/{plugins → extensions}/components/RoomEnvironmentComponent.module.js +0 -0
  147. /package/{plugins → extensions}/components/GridHelperComponent.ts +0 -0
  148. /package/{plugins → extensions}/components/LightHelperComponent.ts +0 -0
  149. /package/{plugins → extensions}/components/RoomEnvironmentComponent.ts +0 -0
  150. /package/{plugins → extensions}/loaders/IFCX/IFCXLoader.ts +0 -0
  151. /package/{plugins → extensions}/loaders/IFCX/index.ts +0 -0
  152. /package/{plugins → extensions}/loaders/IFCX/render.js +0 -0
  153. /package/{plugins → extensions}/loaders/Potree/index.ts +0 -0
@@ -21,17 +21,21 @@
21
21
  // acknowledge and accept the above terms.
22
22
  ///////////////////////////////////////////////////////////////////////////////
23
23
 
24
- import { AxesHelper, Vector3 } from "three";
24
+ import { AxesHelper, Box3, Vector3 } from "three";
25
25
  import { IComponent, components, Viewer } from "@inweb/viewer-three";
26
26
 
27
27
  class AxesHelperComponent implements IComponent {
28
28
  private viewer: Viewer;
29
29
  private axesHelper1: AxesHelper;
30
30
  private axesHelper2: AxesHelper;
31
+ private modelHelpers: AxesHelper[];
31
32
 
32
33
  constructor(viewer: Viewer) {
33
34
  this.axesHelper1 = new AxesHelper(1);
34
35
  this.axesHelper2 = new AxesHelper(1);
36
+ this.modelHelpers = [];
37
+
38
+ this.axesHelper1.setColors("#ccc", "#ccc", "#ccc");
35
39
 
36
40
  this.viewer = viewer;
37
41
  this.viewer.addEventListener("initialize", this.syncHelper);
@@ -40,6 +44,11 @@ class AxesHelperComponent implements IComponent {
40
44
  }
41
45
 
42
46
  dispose() {
47
+ this.modelHelpers.forEach((helper) => {
48
+ helper.removeFromParent();
49
+ helper.dispose();
50
+ });
51
+
43
52
  this.axesHelper1.removeFromParent();
44
53
  this.axesHelper1.dispose();
45
54
 
@@ -52,6 +61,12 @@ class AxesHelperComponent implements IComponent {
52
61
  }
53
62
 
54
63
  syncHelper = () => {
64
+ this.modelHelpers.forEach((helper) => {
65
+ helper.removeFromParent();
66
+ helper.dispose();
67
+ });
68
+ this.modelHelpers.length = 0;
69
+
55
70
  this.axesHelper1.removeFromParent();
56
71
  this.axesHelper2.removeFromParent();
57
72
 
@@ -59,13 +74,27 @@ class AxesHelperComponent implements IComponent {
59
74
  const center = this.viewer.extents.getCenter(new Vector3());
60
75
 
61
76
  this.axesHelper1.position.set(0, 0, 0);
62
- this.axesHelper1.scale.setScalar(size);
77
+ this.axesHelper1.scale.setScalar(size * 1.25);
63
78
 
64
79
  this.axesHelper2.position.copy(center);
65
80
  this.axesHelper2.scale.setScalar(size);
66
81
 
67
82
  this.viewer.helpers.add(this.axesHelper1);
68
83
  this.viewer.helpers.add(this.axesHelper2);
84
+
85
+ if (this.viewer.models.length < 2) return;
86
+
87
+ this.viewer.models.forEach((model) => {
88
+ const extents = model.getExtents(new Box3());
89
+ const size = extents.getSize(new Vector3()).length();
90
+ const center = extents.getCenter(new Vector3());
91
+
92
+ const helper = new AxesHelper(size);
93
+ helper.position.copy(center);
94
+
95
+ this.modelHelpers.push(helper);
96
+ this.viewer.helpers.add(helper);
97
+ });
69
98
  };
70
99
  }
71
100
 
@@ -27,9 +27,12 @@ import { IComponent, components, Viewer } from "@inweb/viewer-three";
27
27
  class ExtentsHelperComponent implements IComponent {
28
28
  private viewer: Viewer;
29
29
  private boxHelper: Box3Helper;
30
+ private modelHelpers: Box3Helper[];
30
31
 
31
32
  constructor(viewer: Viewer) {
32
33
  this.boxHelper = new Box3Helper(new Box3(), "#ff0000");
34
+ this.modelHelpers = [];
35
+
33
36
  this.viewer = viewer;
34
37
  this.viewer.on("geometryend", this.syncHelper);
35
38
  this.viewer.on("clear", this.syncHelper);
@@ -41,6 +44,11 @@ class ExtentsHelperComponent implements IComponent {
41
44
  }
42
45
 
43
46
  dispose() {
47
+ this.modelHelpers.forEach((helper) => {
48
+ helper.removeFromParent();
49
+ helper.dispose();
50
+ });
51
+
44
52
  this.boxHelper.removeFromParent();
45
53
  this.boxHelper.dispose();
46
54
 
@@ -54,12 +62,29 @@ class ExtentsHelperComponent implements IComponent {
54
62
  }
55
63
 
56
64
  syncHelper = () => {
65
+ this.modelHelpers.forEach((helper) => {
66
+ helper.removeFromParent();
67
+ helper.dispose();
68
+ });
69
+ this.modelHelpers.length = 0;
70
+
57
71
  this.boxHelper.removeFromParent();
58
72
 
59
73
  if (this.viewer.extents.isEmpty()) return;
60
74
 
61
75
  this.boxHelper.box = this.viewer.extents.clone();
62
76
  this.viewer.helpers.add(this.boxHelper);
77
+
78
+ if (this.viewer.models.length < 2) return;
79
+
80
+ this.viewer.models.forEach((model) => {
81
+ const extents = model.getExtents(new Box3());
82
+
83
+ const helper = new Box3Helper(extents, "#ff0000");
84
+
85
+ this.modelHelpers.push(helper);
86
+ this.viewer.helpers.add(helper);
87
+ });
63
88
  };
64
89
  }
65
90
 
@@ -0,0 +1,197 @@
1
+ ///////////////////////////////////////////////////////////////////////////////
2
+ // Copyright (C) 2002-2025, Open Design Alliance (the "Alliance").
3
+ // All rights reserved.
4
+ //
5
+ // This software and its documentation and related materials are owned by
6
+ // the Alliance. The software may only be incorporated into application
7
+ // programs owned by members of the Alliance, subject to a signed
8
+ // Membership Agreement and Supplemental Software License Agreement with the
9
+ // Alliance. The structure and organization of this software are the valuable
10
+ // trade secrets of the Alliance and its suppliers. The software is also
11
+ // protected by copyright law and international treaty provisions. Application
12
+ // programs incorporating this software must include the following statement
13
+ // with their copyright notices:
14
+ //
15
+ // This application incorporates Open Design Alliance software pursuant to a
16
+ // license agreement with Open Design Alliance.
17
+ // Open Design Alliance Copyright (C) 2002-2025 by Open Design Alliance.
18
+ // All rights reserved.
19
+ //
20
+ // By use of this software, its documentation or related materials, you
21
+ // acknowledge and accept the above terms.
22
+ ///////////////////////////////////////////////////////////////////////////////
23
+
24
+ import { IComponent, components, Viewer } from "@inweb/viewer-three";
25
+
26
+ const map = {
27
+ B: 1,
28
+ KB: 1 << 10,
29
+ MB: 1 << 20,
30
+ GB: 1 << 30,
31
+ };
32
+
33
+ function formatBytes(bytes: number): string {
34
+ if (!bytes) return "-";
35
+
36
+ let unit: string;
37
+ if (bytes >= map.GB) unit = "GB";
38
+ else if (bytes >= map.MB) unit = "MB";
39
+ else if (bytes >= map.KB) unit = "KB";
40
+ else unit = "B";
41
+
42
+ const value = bytes / map[unit];
43
+ return value.toFixed() + " " + unit;
44
+ }
45
+
46
+ class Panel {
47
+ public dom: HTMLElement;
48
+ private label: HTMLElement;
49
+ private text: HTMLElement;
50
+
51
+ constructor(label: string) {
52
+ this.dom = document.createElement("div");
53
+
54
+ this.label = document.createElement("div");
55
+ this.label.style.padding = "0.25rem 0";
56
+ this.label.style.fontWeight = "600";
57
+ this.label.innerText = label;
58
+ this.dom.appendChild(this.label);
59
+
60
+ this.text = document.createElement("small");
61
+ this.text.style.display = "block";
62
+ this.text.style.padding = "0 0.5rem";
63
+ this.dom.appendChild(this.text);
64
+ }
65
+
66
+ update(text: string) {
67
+ if (this.text.innerText !== text) this.text.innerText = text;
68
+ }
69
+ }
70
+
71
+ class InfoPanelComponent implements IComponent {
72
+ private viewer: Viewer;
73
+ private container: HTMLElement;
74
+ private performancePanel: Panel;
75
+ private renderPanel: Panel;
76
+ private optimizedPanel: Panel;
77
+ private scenePanel: Panel;
78
+ private memoryPanel: Panel;
79
+
80
+ constructor(viewer: Viewer) {
81
+ this.container = document.createElement("div");
82
+ this.container.id = "info-container";
83
+ this.container.style.position = "absolute";
84
+ this.container.style.left = "0px";
85
+ this.container.style.top = "0px";
86
+ this.container.style.maxHeight = "100%";
87
+ this.container.style.overflow = "auto";
88
+ this.container.style.padding = "1rem";
89
+
90
+ this.setTheme("dark");
91
+
92
+ this.performancePanel = new Panel("Performance");
93
+ this.renderPanel = new Panel("Render");
94
+ this.optimizedPanel = new Panel("Optimized Scene");
95
+ this.scenePanel = new Panel("Scene");
96
+ this.memoryPanel = new Panel("Memory");
97
+
98
+ this.container.appendChild(this.performancePanel.dom);
99
+ this.container.appendChild(this.renderPanel.dom);
100
+ this.container.appendChild(this.optimizedPanel.dom);
101
+ this.container.appendChild(this.scenePanel.dom);
102
+ this.container.appendChild(this.memoryPanel.dom);
103
+
104
+ viewer.canvas.parentElement.appendChild(this.container);
105
+
106
+ this.viewer = viewer;
107
+ this.viewer.addEventListener("clear", this.updateSceneInfo);
108
+ this.viewer.addEventListener("geometryend", this.updateSceneInfo);
109
+ this.viewer.addEventListener("render", this.updateRenderInfo);
110
+ this.viewer.addEventListener("animate", this.updatePreformanceInfo);
111
+
112
+ this.updatePreformanceInfo();
113
+ this.updateRenderInfo();
114
+ this.updateSceneInfo();
115
+ }
116
+
117
+ dispose() {
118
+ this.viewer.removeEventListener("clear", this.updateSceneInfo);
119
+ this.viewer.removeEventListener("geometryend", this.updateSceneInfo);
120
+ this.viewer.removeEventListener("render", this.updateRenderInfo);
121
+ this.viewer.removeEventListener("animate", this.updatePreformanceInfo);
122
+
123
+ this.performancePanel = undefined;
124
+ this.renderPanel = undefined;
125
+ this.optimizedPanel = undefined;
126
+ this.scenePanel = undefined;
127
+ this.memoryPanel = undefined;
128
+
129
+ this.container.remove();
130
+ this.container = undefined;
131
+ }
132
+
133
+ setTheme(value: string) {
134
+ if (value === "light") {
135
+ this.container.style.background = "rgba(0, 0, 0, 0.025)";
136
+ this.container.style.color = "#3d3d3d";
137
+ }
138
+ if (value === "dark") {
139
+ this.container.style.background = "rgba(0, 0, 0, 0.88)";
140
+ this.container.style.color = "#ebebeb";
141
+ }
142
+ }
143
+
144
+ updatePreformanceInfo = () => {
145
+ const info = this.viewer.info;
146
+
147
+ const text = [];
148
+ text.push(`FPS: ${info.performance.fps}`);
149
+ text.push(`Frame Time: ${info.performance.frameTime} ms`);
150
+ this.performancePanel.update(text.join("\n"));
151
+
152
+ this.viewer.update();
153
+ };
154
+
155
+ updateRenderInfo = () => {
156
+ const info = this.viewer.info;
157
+
158
+ const text = [];
159
+ text.push(`Viewport: ${info.render.viewport.width} x ${info.render.viewport.height}`);
160
+ text.push(`Antialiasing: ${info.render.antialiasing}`);
161
+ text.push(`Draw Calls: ${info.render.drawCalls}`);
162
+ text.push(`Triangles: ${info.render.triangles}`);
163
+ text.push(`Points: ${info.render.points}`);
164
+ text.push(`Lines: ${info.render.lines}`);
165
+ this.renderPanel.update(text.join("\n"));
166
+ };
167
+
168
+ updateSceneInfo = () => {
169
+ const info = this.viewer.info;
170
+
171
+ const text = [];
172
+ text.push(`Objects: ${info.optimizedScene.objects}`);
173
+ text.push(`Triangles: ${info.optimizedScene.triangles}`);
174
+ text.push(`Points: ${info.optimizedScene.points}`);
175
+ text.push(`Lines: ${info.optimizedScene.lines}`);
176
+ text.push(`Edges: ${info.optimizedScene.edges}`);
177
+ this.optimizedPanel.update(text.join("\n"));
178
+
179
+ text.length = 0;
180
+ text.push(`Objects: ${info.scene.objects}`);
181
+ text.push(`Triangles: ${info.scene.triangles}`);
182
+ text.push(`Points: ${info.scene.points}`);
183
+ text.push(`Lines: ${info.scene.lines}`);
184
+ text.push(`Edges: ${info.scene.edges}`);
185
+ this.scenePanel.update(text.join("\n"));
186
+
187
+ text.length = 0;
188
+ text.push(`Geometries: ${info.memory.geometries}`);
189
+ text.push(`Textures: ${info.memory.textures}`);
190
+ text.push(`Materials: ${info.memory.materials}`);
191
+ text.push(`GPU Used: ${formatBytes(info.memory.totalEstimatedGpuBytes)}`);
192
+ text.push(`JS Heap Used: ${formatBytes(info.memory.usedJSHeapSize)}`);
193
+ this.memoryPanel.update(text.join("\n"));
194
+ };
195
+ }
196
+
197
+ components.registerComponent("InfoPanelComponent", (viewer) => new InfoPanelComponent(viewer));
@@ -31,23 +31,30 @@ class StatsPanelComponent implements IComponent {
31
31
  constructor(viewer: Viewer) {
32
32
  this.stats = new Stats();
33
33
  this.stats.dom.style.position = "absolute";
34
+ this.stats.dom.style.left = "0px";
35
+ this.stats.dom.style.top = "0px";
34
36
  viewer.canvas.parentElement.appendChild(this.stats.dom);
35
37
 
36
38
  this.viewer = viewer;
37
- this.viewer.on("animate", this.updateStats);
39
+ this.viewer.addEventListener("render", this.updateStats);
40
+ this.viewer.addEventListener("animate", this.updateViewer);
38
41
  }
39
42
 
40
43
  dispose() {
41
- this.viewer.off("animate", this.updateStats);
44
+ this.viewer.removeEventListener("render", this.updateStats);
45
+ this.viewer.removeEventListener("animate", this.updateViewer);
42
46
 
43
47
  this.stats.dom.remove();
44
48
  this.stats = undefined;
45
49
  }
46
50
 
47
51
  updateStats = () => {
48
- this.viewer.render(null, true);
49
52
  this.stats.update();
50
53
  };
54
+
55
+ updateViewer = () => {
56
+ this.viewer.update();
57
+ };
51
58
  }
52
59
 
53
60
  components.registerComponent("StatsPanelComponent", (viewer) => new StatsPanelComponent(viewer));
@@ -58,10 +58,9 @@ class GLTFCloudLoader extends Loader {
58
58
  if (!this.viewer.scene) return this;
59
59
 
60
60
  const modelImpl = new ModelImpl(gltf.scene);
61
- modelImpl.loader = this;
62
- modelImpl.viewer = this.viewer;
61
+ modelImpl.id = model.file.id;
63
62
 
64
- this.viewer.scene.add(modelImpl.scene);
63
+ this.viewer.scene.add(gltf.scene);
65
64
  this.viewer.models.push(modelImpl);
66
65
 
67
66
  this.viewer.syncOptions();
@@ -22,20 +22,21 @@
22
22
  ///////////////////////////////////////////////////////////////////////////////
23
23
 
24
24
  import { GLTFLoader } from "three/examples/jsm/loaders/GLTFLoader.js";
25
- import { Loader } from "@inweb/viewer-core";
26
-
27
- import { Viewer } from "../Viewer";
28
- import { GLTFLoadingManager, GLTFLoadParams } from "./GLTFLoadingManager";
29
- import { ModelImpl } from "../models/ModelImpl";
25
+ import { GLTFLoadingManager, GLTFLoadParams, Loader, loaders, ModelImpl, Viewer } from "@inweb/viewer-three";
30
26
 
31
27
  export class GLTFFileLoader extends Loader {
32
28
  public viewer: Viewer;
29
+ private manager: GLTFLoadingManager;
33
30
 
34
31
  constructor(viewer: Viewer) {
35
32
  super();
36
33
  this.viewer = viewer;
37
34
  }
38
35
 
36
+ override dispose() {
37
+ if (this.manager) this.manager.dispose();
38
+ }
39
+
39
40
  override isSupport(file: any, format?: string): boolean {
40
41
  return (
41
42
  (typeof file === "string" || file instanceof globalThis.File || file instanceof ArrayBuffer) &&
@@ -43,12 +44,13 @@ export class GLTFFileLoader extends Loader {
43
44
  );
44
45
  }
45
46
 
46
- override async load(file: any, format?: string, params?: GLTFLoadParams): Promise<this> {
47
- const manager = new GLTFLoadingManager(file, params);
47
+ override async load(file: any, format?: string, params: GLTFLoadParams = {}): Promise<this> {
48
+ this.manager = new GLTFLoadingManager(file, params);
48
49
 
49
- const loader = new GLTFLoader(manager);
50
- loader.setPath(manager.path);
50
+ const loader = new GLTFLoader(this.manager);
51
+ loader.setPath(this.manager.path);
51
52
  loader.setCrossOrigin(params.crossOrigin || loader.crossOrigin);
53
+ loader.setRequestHeader((params.requestHeader as any) || {});
52
54
  loader.setWithCredentials(params.withCredentials || loader.withCredentials);
53
55
 
54
56
  const progress = (event: ProgressEvent) => {
@@ -57,12 +59,17 @@ export class GLTFFileLoader extends Loader {
57
59
  this.viewer.emitEvent({ type: "geometryprogress", data: progress, file });
58
60
  };
59
61
 
60
- const gltf = await loader.loadAsync(manager.fileURL, progress);
62
+ const gltf = await loader.loadAsync(this.manager.fileURL, progress);
61
63
  if (!this.viewer.scene) return this;
62
64
 
65
+ let handle = 0;
66
+ gltf.scene.traverse((object) => {
67
+ object.userData = { handle: handle + "", ...object.userData };
68
+ handle++;
69
+ });
70
+
63
71
  const modelImpl = new ModelImpl(gltf.scene);
64
- modelImpl.loader = this;
65
- modelImpl.viewer = this.viewer;
72
+ modelImpl.id = params.modelId || this.extractFileName(file);
66
73
 
67
74
  this.viewer.scene.add(gltf.scene);
68
75
  this.viewer.models.push(modelImpl);
@@ -76,3 +83,5 @@ export class GLTFFileLoader extends Loader {
76
83
  return this;
77
84
  }
78
85
  }
86
+
87
+ loaders.registerLoader("gltf-file", (viewer: any) => new GLTFFileLoader(viewer));
@@ -21,6 +21,7 @@
21
21
  // acknowledge and accept the above terms.
22
22
  ///////////////////////////////////////////////////////////////////////////////
23
23
 
24
+ import { Scene } from "three";
24
25
  import { Loader, ModelImpl, Viewer } from "@inweb/viewer-three";
25
26
  import { parse, clear } from "./render.js";
26
27
 
@@ -52,18 +53,17 @@ export class IFCXCloudLoader extends Loader {
52
53
  const textDecoder = new TextDecoder();
53
54
  const json = JSON.parse(textDecoder.decode(arrayBuffer));
54
55
 
55
- const scene = await parse(json);
56
+ const scene: Scene = await parse(json);
56
57
  clear();
57
58
 
58
59
  let handle = 0;
59
- scene.traverse((object: any) => {
60
- object.userData = { handle, ...object.userData };
60
+ scene.traverse((object) => {
61
+ object.userData = { handle: handle + "", ...object.userData };
61
62
  handle++;
62
63
  });
63
64
 
64
65
  const modelImpl = new ModelImpl(scene);
65
- modelImpl.loader = this;
66
- modelImpl.viewer = this.viewer;
66
+ modelImpl.id = file.id;
67
67
 
68
68
  this.viewer.scene.add(scene);
69
69
  this.viewer.models.push(modelImpl);
@@ -39,7 +39,7 @@ export class IFCXFileLoader extends Loader {
39
39
  );
40
40
  }
41
41
 
42
- override async load(file: any, format?: string, params?: GLTFLoadParams): Promise<this> {
42
+ override async load(file: any, format?: string, params: GLTFLoadParams = {}): Promise<this> {
43
43
  const manager = new GLTFLoadingManager(file, params);
44
44
 
45
45
  const loader = new IFCXLoader(manager);
@@ -58,13 +58,12 @@ export class IFCXFileLoader extends Loader {
58
58
 
59
59
  let handle = 0;
60
60
  scene.traverse((object) => {
61
- object.userData = { handle, ...object.userData };
61
+ object.userData = { handle: handle + "", ...object.userData };
62
62
  handle++;
63
63
  });
64
64
 
65
65
  const modelImpl = new ModelImpl(scene);
66
- modelImpl.loader = this;
67
- modelImpl.viewer = this.viewer;
66
+ modelImpl.id = params.modelId || this.extractFileName(file);
68
67
 
69
68
  this.viewer.scene.add(scene);
70
69
  this.viewer.models.push(modelImpl);
@@ -23,11 +23,11 @@
23
23
 
24
24
  import { Euler, Vector3 } from "three";
25
25
  import { PointColorType, PointCloudOctree, PointSizeType, PointShape, Potree } from "potree-core";
26
- import { Loader, Viewer } from "@inweb/viewer-three";
26
+ import { Loader, LoadParams, Viewer } from "@inweb/viewer-three";
27
27
 
28
28
  import { PotreeModelImpl } from "./PotreeModelImpl";
29
29
 
30
- export type PotreeLoadParams = {
30
+ export type PotreeLoadParams = LoadParams & {
31
31
  path?: string;
32
32
  position?: Vector3;
33
33
  rotation?: Euler;
@@ -83,8 +83,7 @@ export class PotreeFileLoader extends Loader {
83
83
  this.pointClouds.push(pco);
84
84
 
85
85
  const modelImpl = new PotreeModelImpl(pco);
86
- modelImpl.loader = this;
87
- modelImpl.viewer = this.viewer;
86
+ modelImpl.id = params.modelId || this.extractFileName(file);
88
87
  modelImpl.pco = pco;
89
88
 
90
89
  this.viewer.scene.add(pco);
@@ -0,0 +1,108 @@
1
+ ///////////////////////////////////////////////////////////////////////////////
2
+ // Copyright (C) 2002-2025, Open Design Alliance (the "Alliance").
3
+ // All rights reserved.
4
+ //
5
+ // This software and its documentation and related materials are owned by
6
+ // the Alliance. The software may only be incorporated into application
7
+ // programs owned by members of the Alliance, subject to a signed
8
+ // Membership Agreement and Supplemental Software License Agreement with the
9
+ // Alliance. The structure and organization of this software are the valuable
10
+ // trade secrets of the Alliance and its suppliers. The software is also
11
+ // protected by copyright law and international treaty provisions. Application
12
+ // programs incorporating this software must include the following statement
13
+ // with their copyright notices:
14
+ //
15
+ // This application incorporates Open Design Alliance software pursuant to a
16
+ // license agreement with Open Design Alliance.
17
+ // Open Design Alliance Copyright (C) 2002-2025 by Open Design Alliance.
18
+ // All rights reserved.
19
+ //
20
+ // By use of this software, its documentation or related materials, you
21
+ // acknowledge and accept the above terms.
22
+ ///////////////////////////////////////////////////////////////////////////////
23
+
24
+ import { Box3 } from "three";
25
+ import { PointCloudOctree } from "potree-core";
26
+ import { IInfo, Info } from "@inweb/viewer-core";
27
+ import { ModelImpl } from "@inweb/viewer-three";
28
+
29
+ // Potree model implementation.
30
+
31
+ export class PotreeModelImpl extends ModelImpl {
32
+ public pco: PointCloudOctree;
33
+
34
+ override getInfo(): IInfo {
35
+ // ===================== AI-CODE-START ======================
36
+ // Source: Claude Sonnet 4.5
37
+ // Date: 2025-10-02
38
+ // Reviewer: roman.mochalov@opendesign.com
39
+ // Issue: CLOUD-5738
40
+
41
+ let totalPoints = 0;
42
+ let totalNodes = 0;
43
+ let loadedGeometryBytes = 0;
44
+ let estimatedTotalGeometryBytes = 0;
45
+ let geometryBytes = 0;
46
+
47
+ const bytesPerPoint = this.pco.pcoGeometry?.pointAttributes?.byteSize || 0;
48
+
49
+ if (this.pco.pcoGeometry && this.pco.pcoGeometry.root) {
50
+ this.pco.pcoGeometry.root.traverse((node: any) => {
51
+ totalNodes++;
52
+
53
+ const numPoints = node.numPoints || 0;
54
+ totalPoints += numPoints;
55
+
56
+ if (node.loaded && node.geometry) {
57
+ if (node.geometry.attributes) {
58
+ for (const name in node.geometry.attributes) {
59
+ const attribute = node.geometry.attributes[name];
60
+ if (attribute && attribute.array) {
61
+ loadedGeometryBytes += attribute.array.byteLength;
62
+ }
63
+ }
64
+ }
65
+
66
+ if (node.geometry.index && node.geometry.index.array) {
67
+ loadedGeometryBytes += node.geometry.index.array.byteLength;
68
+ }
69
+ }
70
+
71
+ if (bytesPerPoint > 0 && numPoints > 0) {
72
+ estimatedTotalGeometryBytes += numPoints * (bytesPerPoint * 2 - 1);
73
+ }
74
+ }, true);
75
+ }
76
+
77
+ geometryBytes = Math.max(loadedGeometryBytes, estimatedTotalGeometryBytes);
78
+
79
+ // ===================== AI-CODE-END ======================
80
+
81
+ const info = new Info();
82
+
83
+ info.scene.objects = totalNodes;
84
+ info.scene.points = totalPoints;
85
+ info.scene.triangles = 0;
86
+ info.scene.lines = 0;
87
+ info.scene.edges = 0;
88
+
89
+ info.memory.geometries = totalNodes;
90
+ info.memory.geometryBytes = Math.floor(geometryBytes);
91
+ info.memory.textures = 0;
92
+ info.memory.textureBytes = 0;
93
+ info.memory.materials = 1;
94
+ info.memory.totalEstimatedGpuBytes = Math.floor(geometryBytes);
95
+
96
+ info.optimizedScene.objects = info.scene.objects;
97
+ info.optimizedScene.triangles = info.scene.triangles;
98
+ info.optimizedScene.points = info.scene.points;
99
+ info.optimizedScene.lines = info.scene.lines;
100
+ info.optimizedScene.edges = info.scene.edges;
101
+
102
+ return info;
103
+ }
104
+
105
+ override getExtents(target: Box3): Box3 {
106
+ return target.union(this.pco.pcoGeometry.boundingBox);
107
+ }
108
+ }