@combeenation/3d-viewer 12.4.1 → 13.0.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 (213) hide show
  1. package/dist/lib-cjs/buildinfo.json +1 -1
  2. package/dist/lib-cjs/commonjs.tsconfig.tsbuildinfo +1 -1
  3. package/dist/lib-cjs/index.d.ts +63 -63
  4. package/dist/lib-cjs/index.js +81 -114
  5. package/dist/lib-cjs/index.js.map +1 -1
  6. package/dist/lib-cjs/internal/cbnCustomBabylonLoaderPlugin.d.ts +10 -0
  7. package/dist/lib-cjs/internal/cbnCustomBabylonLoaderPlugin.js +124 -0
  8. package/dist/lib-cjs/internal/cbnCustomBabylonLoaderPlugin.js.map +1 -0
  9. package/dist/lib-cjs/internal/cloningHelper.d.ts +19 -0
  10. package/dist/lib-cjs/internal/cloningHelper.js +165 -0
  11. package/dist/lib-cjs/internal/cloningHelper.js.map +1 -0
  12. package/dist/lib-cjs/internal/deviceHelper.d.ts +9 -0
  13. package/dist/lib-cjs/{api/util → internal}/deviceHelper.js +6 -10
  14. package/dist/lib-cjs/internal/deviceHelper.js.map +1 -0
  15. package/dist/lib-cjs/internal/geometryHelper.d.ts +21 -0
  16. package/dist/lib-cjs/{api/util → internal}/geometryHelper.js +57 -24
  17. package/dist/lib-cjs/internal/geometryHelper.js.map +1 -0
  18. package/dist/lib-cjs/internal/metadataHelper.d.ts +26 -0
  19. package/dist/lib-cjs/internal/metadataHelper.js +51 -0
  20. package/dist/lib-cjs/internal/metadataHelper.js.map +1 -0
  21. package/dist/lib-cjs/internal/paintableHelper.d.ts +40 -0
  22. package/dist/lib-cjs/internal/paintableHelper.js +287 -0
  23. package/dist/lib-cjs/internal/paintableHelper.js.map +1 -0
  24. package/dist/lib-cjs/internal/tagsHelper.d.ts +12 -0
  25. package/dist/lib-cjs/internal/tagsHelper.js +38 -0
  26. package/dist/lib-cjs/internal/tagsHelper.js.map +1 -0
  27. package/dist/lib-cjs/manager/cameraManager.d.ts +51 -0
  28. package/dist/lib-cjs/manager/cameraManager.js +154 -0
  29. package/dist/lib-cjs/manager/cameraManager.js.map +1 -0
  30. package/dist/lib-cjs/manager/debugManager.d.ts +60 -0
  31. package/dist/lib-cjs/manager/debugManager.js +218 -0
  32. package/dist/lib-cjs/manager/debugManager.js.map +1 -0
  33. package/dist/lib-cjs/manager/eventManager.d.ts +52 -0
  34. package/dist/lib-cjs/manager/eventManager.js +72 -0
  35. package/dist/lib-cjs/manager/eventManager.js.map +1 -0
  36. package/dist/lib-cjs/{api/manager → manager}/gltfExportManager.d.ts +29 -34
  37. package/dist/lib-cjs/{api/manager → manager}/gltfExportManager.js +99 -120
  38. package/dist/lib-cjs/manager/gltfExportManager.js.map +1 -0
  39. package/dist/lib-cjs/manager/materialManager.d.ts +35 -0
  40. package/dist/lib-cjs/manager/materialManager.js +126 -0
  41. package/dist/lib-cjs/manager/materialManager.js.map +1 -0
  42. package/dist/lib-cjs/manager/modelManager.d.ts +145 -0
  43. package/dist/lib-cjs/manager/modelManager.js +381 -0
  44. package/dist/lib-cjs/manager/modelManager.js.map +1 -0
  45. package/dist/lib-cjs/manager/parameterManager.d.ts +210 -0
  46. package/dist/lib-cjs/manager/parameterManager.js +515 -0
  47. package/dist/lib-cjs/manager/parameterManager.js.map +1 -0
  48. package/dist/lib-cjs/manager/sceneManager.d.ts +45 -0
  49. package/dist/lib-cjs/manager/sceneManager.js +65 -0
  50. package/dist/lib-cjs/manager/sceneManager.js.map +1 -0
  51. package/dist/lib-cjs/manager/screenshotManager.d.ts +36 -0
  52. package/dist/lib-cjs/manager/screenshotManager.js +40 -0
  53. package/dist/lib-cjs/manager/screenshotManager.js.map +1 -0
  54. package/dist/lib-cjs/manager/textureManager.d.ts +12 -0
  55. package/dist/lib-cjs/manager/textureManager.js +44 -0
  56. package/dist/lib-cjs/manager/textureManager.js.map +1 -0
  57. package/dist/lib-cjs/viewer.d.ts +117 -0
  58. package/dist/lib-cjs/viewer.js +222 -0
  59. package/dist/lib-cjs/viewer.js.map +1 -0
  60. package/dist/lib-cjs/{api/classes/viewerError.d.ts → viewerError.d.ts} +6 -1
  61. package/dist/lib-cjs/{api/classes/viewerError.js → viewerError.js} +6 -1
  62. package/dist/lib-cjs/viewerError.js.map +1 -0
  63. package/package.json +10 -11
  64. package/src/dev.ts +14 -37
  65. package/src/{types.d.ts → globalTypes.d.ts} +8 -18
  66. package/src/index.ts +79 -113
  67. package/src/internal/cbnCustomBabylonLoaderPlugin.ts +149 -0
  68. package/src/internal/cloningHelper.ts +225 -0
  69. package/src/internal/deviceHelper.ts +25 -0
  70. package/src/{api/util → internal}/geometryHelper.ts +63 -24
  71. package/src/internal/metadataHelper.ts +63 -0
  72. package/src/internal/paintableHelper.ts +310 -0
  73. package/src/internal/tagsHelper.ts +41 -0
  74. package/src/manager/cameraManager.ts +236 -0
  75. package/src/manager/debugManager.ts +245 -0
  76. package/src/manager/eventManager.ts +72 -0
  77. package/src/{api/manager → manager}/gltfExportManager.ts +132 -125
  78. package/src/manager/materialManager.ts +135 -0
  79. package/src/manager/modelManager.ts +456 -0
  80. package/src/manager/parameterManager.ts +652 -0
  81. package/src/manager/sceneManager.ts +101 -0
  82. package/src/manager/screenshotManager.ts +59 -0
  83. package/src/manager/textureManager.ts +32 -0
  84. package/src/viewer.ts +296 -0
  85. package/src/{api/classes/viewerError.ts → viewerError.ts} +6 -1
  86. package/dist/lib-cjs/api/classes/animationInterface.d.ts +0 -8
  87. package/dist/lib-cjs/api/classes/animationInterface.js +0 -3
  88. package/dist/lib-cjs/api/classes/animationInterface.js.map +0 -1
  89. package/dist/lib-cjs/api/classes/dottedPath.d.ts +0 -79
  90. package/dist/lib-cjs/api/classes/dottedPath.js +0 -167
  91. package/dist/lib-cjs/api/classes/dottedPath.js.map +0 -1
  92. package/dist/lib-cjs/api/classes/element.d.ts +0 -153
  93. package/dist/lib-cjs/api/classes/element.js +0 -703
  94. package/dist/lib-cjs/api/classes/element.js.map +0 -1
  95. package/dist/lib-cjs/api/classes/event.d.ts +0 -401
  96. package/dist/lib-cjs/api/classes/event.js +0 -425
  97. package/dist/lib-cjs/api/classes/event.js.map +0 -1
  98. package/dist/lib-cjs/api/classes/eventBroadcaster.d.ts +0 -26
  99. package/dist/lib-cjs/api/classes/eventBroadcaster.js +0 -50
  100. package/dist/lib-cjs/api/classes/eventBroadcaster.js.map +0 -1
  101. package/dist/lib-cjs/api/classes/fuzzyMap.d.ts +0 -7
  102. package/dist/lib-cjs/api/classes/fuzzyMap.js +0 -22
  103. package/dist/lib-cjs/api/classes/fuzzyMap.js.map +0 -1
  104. package/dist/lib-cjs/api/classes/parameter.d.ts +0 -410
  105. package/dist/lib-cjs/api/classes/parameter.js +0 -643
  106. package/dist/lib-cjs/api/classes/parameter.js.map +0 -1
  107. package/dist/lib-cjs/api/classes/parameterObservable.d.ts +0 -36
  108. package/dist/lib-cjs/api/classes/parameterObservable.js +0 -73
  109. package/dist/lib-cjs/api/classes/parameterObservable.js.map +0 -1
  110. package/dist/lib-cjs/api/classes/parameterizable.d.ts +0 -15
  111. package/dist/lib-cjs/api/classes/parameterizable.js +0 -103
  112. package/dist/lib-cjs/api/classes/parameterizable.js.map +0 -1
  113. package/dist/lib-cjs/api/classes/placementAnimation.d.ts +0 -45
  114. package/dist/lib-cjs/api/classes/placementAnimation.js +0 -177
  115. package/dist/lib-cjs/api/classes/placementAnimation.js.map +0 -1
  116. package/dist/lib-cjs/api/classes/variant.d.ts +0 -261
  117. package/dist/lib-cjs/api/classes/variant.js +0 -873
  118. package/dist/lib-cjs/api/classes/variant.js.map +0 -1
  119. package/dist/lib-cjs/api/classes/variantInstance.d.ts +0 -53
  120. package/dist/lib-cjs/api/classes/variantInstance.js +0 -126
  121. package/dist/lib-cjs/api/classes/variantInstance.js.map +0 -1
  122. package/dist/lib-cjs/api/classes/variantParameterizable.d.ts +0 -17
  123. package/dist/lib-cjs/api/classes/variantParameterizable.js +0 -87
  124. package/dist/lib-cjs/api/classes/variantParameterizable.js.map +0 -1
  125. package/dist/lib-cjs/api/classes/viewer.d.ts +0 -215
  126. package/dist/lib-cjs/api/classes/viewer.js +0 -709
  127. package/dist/lib-cjs/api/classes/viewer.js.map +0 -1
  128. package/dist/lib-cjs/api/classes/viewerError.js.map +0 -1
  129. package/dist/lib-cjs/api/classes/viewerLight.d.ts +0 -66
  130. package/dist/lib-cjs/api/classes/viewerLight.js +0 -345
  131. package/dist/lib-cjs/api/classes/viewerLight.js.map +0 -1
  132. package/dist/lib-cjs/api/internal/lensRendering.d.ts +0 -8
  133. package/dist/lib-cjs/api/internal/lensRendering.js +0 -12
  134. package/dist/lib-cjs/api/internal/lensRendering.js.map +0 -1
  135. package/dist/lib-cjs/api/internal/sceneSetup.d.ts +0 -13
  136. package/dist/lib-cjs/api/internal/sceneSetup.js +0 -228
  137. package/dist/lib-cjs/api/internal/sceneSetup.js.map +0 -1
  138. package/dist/lib-cjs/api/manager/animationManager.d.ts +0 -30
  139. package/dist/lib-cjs/api/manager/animationManager.js +0 -127
  140. package/dist/lib-cjs/api/manager/animationManager.js.map +0 -1
  141. package/dist/lib-cjs/api/manager/gltfExportManager.js.map +0 -1
  142. package/dist/lib-cjs/api/manager/sceneManager.d.ts +0 -33
  143. package/dist/lib-cjs/api/manager/sceneManager.js +0 -129
  144. package/dist/lib-cjs/api/manager/sceneManager.js.map +0 -1
  145. package/dist/lib-cjs/api/manager/tagManager.d.ts +0 -118
  146. package/dist/lib-cjs/api/manager/tagManager.js +0 -531
  147. package/dist/lib-cjs/api/manager/tagManager.js.map +0 -1
  148. package/dist/lib-cjs/api/manager/textureLoadManager.d.ts +0 -22
  149. package/dist/lib-cjs/api/manager/textureLoadManager.js +0 -108
  150. package/dist/lib-cjs/api/manager/textureLoadManager.js.map +0 -1
  151. package/dist/lib-cjs/api/manager/variantInstanceManager.d.ts +0 -106
  152. package/dist/lib-cjs/api/manager/variantInstanceManager.js +0 -291
  153. package/dist/lib-cjs/api/manager/variantInstanceManager.js.map +0 -1
  154. package/dist/lib-cjs/api/store/specStorage.d.ts +0 -32
  155. package/dist/lib-cjs/api/store/specStorage.js +0 -66
  156. package/dist/lib-cjs/api/store/specStorage.js.map +0 -1
  157. package/dist/lib-cjs/api/util/babylonHelper.d.ts +0 -238
  158. package/dist/lib-cjs/api/util/babylonHelper.js +0 -826
  159. package/dist/lib-cjs/api/util/babylonHelper.js.map +0 -1
  160. package/dist/lib-cjs/api/util/debugHelper.d.ts +0 -9
  161. package/dist/lib-cjs/api/util/debugHelper.js +0 -94
  162. package/dist/lib-cjs/api/util/debugHelper.js.map +0 -1
  163. package/dist/lib-cjs/api/util/deviceHelper.d.ts +0 -9
  164. package/dist/lib-cjs/api/util/deviceHelper.js.map +0 -1
  165. package/dist/lib-cjs/api/util/geometryHelper.d.ts +0 -17
  166. package/dist/lib-cjs/api/util/geometryHelper.js.map +0 -1
  167. package/dist/lib-cjs/api/util/globalTypes.d.ts +0 -490
  168. package/dist/lib-cjs/api/util/globalTypes.js +0 -2
  169. package/dist/lib-cjs/api/util/globalTypes.js.map +0 -1
  170. package/dist/lib-cjs/api/util/resourceHelper.d.ts +0 -58
  171. package/dist/lib-cjs/api/util/resourceHelper.js +0 -215
  172. package/dist/lib-cjs/api/util/resourceHelper.js.map +0 -1
  173. package/dist/lib-cjs/api/util/sceneLoaderHelper.d.ts +0 -58
  174. package/dist/lib-cjs/api/util/sceneLoaderHelper.js +0 -229
  175. package/dist/lib-cjs/api/util/sceneLoaderHelper.js.map +0 -1
  176. package/dist/lib-cjs/api/util/stringHelper.d.ts +0 -13
  177. package/dist/lib-cjs/api/util/stringHelper.js +0 -33
  178. package/dist/lib-cjs/api/util/stringHelper.js.map +0 -1
  179. package/dist/lib-cjs/api/util/structureHelper.d.ts +0 -9
  180. package/dist/lib-cjs/api/util/structureHelper.js +0 -58
  181. package/dist/lib-cjs/api/util/structureHelper.js.map +0 -1
  182. package/src/api/classes/animationInterface.ts +0 -10
  183. package/src/api/classes/dottedPath.ts +0 -181
  184. package/src/api/classes/element.ts +0 -766
  185. package/src/api/classes/event.ts +0 -457
  186. package/src/api/classes/eventBroadcaster.ts +0 -52
  187. package/src/api/classes/fuzzyMap.ts +0 -21
  188. package/src/api/classes/parameter.ts +0 -686
  189. package/src/api/classes/parameterObservable.ts +0 -73
  190. package/src/api/classes/parameterizable.ts +0 -87
  191. package/src/api/classes/placementAnimation.ts +0 -162
  192. package/src/api/classes/variant.ts +0 -965
  193. package/src/api/classes/variantInstance.ts +0 -123
  194. package/src/api/classes/variantParameterizable.ts +0 -83
  195. package/src/api/classes/viewer.ts +0 -751
  196. package/src/api/classes/viewerLight.ts +0 -335
  197. package/src/api/internal/debugViewer.ts +0 -90
  198. package/src/api/internal/lensRendering.ts +0 -9
  199. package/src/api/internal/sceneSetup.ts +0 -208
  200. package/src/api/manager/animationManager.ts +0 -143
  201. package/src/api/manager/sceneManager.ts +0 -134
  202. package/src/api/manager/tagManager.ts +0 -572
  203. package/src/api/manager/textureLoadManager.ts +0 -107
  204. package/src/api/manager/variantInstanceManager.ts +0 -306
  205. package/src/api/store/specStorage.ts +0 -68
  206. package/src/api/util/babylonHelper.ts +0 -915
  207. package/src/api/util/debugHelper.ts +0 -121
  208. package/src/api/util/deviceHelper.ts +0 -31
  209. package/src/api/util/globalTypes.ts +0 -566
  210. package/src/api/util/resourceHelper.ts +0 -201
  211. package/src/api/util/sceneLoaderHelper.ts +0 -247
  212. package/src/api/util/stringHelper.ts +0 -30
  213. package/src/api/util/structureHelper.ts +0 -62
@@ -0,0 +1,652 @@
1
+ import {
2
+ AbstractMesh,
3
+ AbstractScene,
4
+ Color3,
5
+ Material,
6
+ PBRMaterial,
7
+ StandardMaterial,
8
+ TransformNode,
9
+ Vector3,
10
+ Viewer,
11
+ ViewerError,
12
+ ViewerErrorIds,
13
+ } from '../index';
14
+ import { getInternalMetadataValue, setInternalMetadataValue } from '../internal/metadataHelper';
15
+ import { paintableParameterObserver } from '../internal/paintableHelper';
16
+ import { getTags, hasTag } from '../internal/tagsHelper';
17
+ import { capitalize, isString } from 'lodash-es';
18
+
19
+ /**
20
+ * Parameters with a built in observer implementation
21
+ */
22
+ export const BuiltInParameter = {
23
+ Visible: 'visible',
24
+ Material: 'material',
25
+ Position: 'position',
26
+ Rotation: 'rotation',
27
+ Scaling: 'scaling',
28
+ Color: 'color',
29
+ Roughness: 'roughness',
30
+ Metallic: 'metallic',
31
+ Paintable: 'paintable',
32
+ };
33
+
34
+ export type ParameterName = string;
35
+ export type ParameterValue = string | number | boolean;
36
+
37
+ export type TagParameterSubject = { tagName: string; nodeName?: never; materialName?: never };
38
+ export type NodeParameterSubject = { tagName?: never; nodeName: string; materialName?: never };
39
+ export type MaterialParameterSubject = { tagName?: never; nodeName?: never; materialName: string };
40
+
41
+ /**
42
+ * Defines which objects are affected by the parameter.\
43
+ * This can be single nodes and materials or tags, which can be used to apply a parameter to a group of nodes or
44
+ * materials.
45
+ */
46
+ export type ParameterSubject = TagParameterSubject | NodeParameterSubject | MaterialParameterSubject;
47
+
48
+ const isTagParameterSubject = (subject: ParameterSubject): subject is TagParameterSubject =>
49
+ 'tagName' in subject && !!subject.tagName;
50
+ const isNodeParameterSubject = (subject: ParameterSubject): subject is NodeParameterSubject =>
51
+ 'nodeName' in subject && !!subject.nodeName;
52
+ const isMaterialParameterSubject = (subject: ParameterSubject): subject is MaterialParameterSubject =>
53
+ 'materialName' in subject && !!subject.materialName;
54
+
55
+ /**
56
+ * Bulk of targeted parameter values, mainly used as input for {@link ParameterManager.setParameterValues} function.\
57
+ * E.g.
58
+ * ```
59
+ * [
60
+ * { nodeName: 'someMesh', parameterName: BuiltInParameter.Visible, value: false },
61
+ * { materialName: 'someMaterial', parameterName: BuiltInParameter.Color, value: '#DD0060' }
62
+ * ]
63
+ * ```
64
+ * Each parameter value entry has to set exactly one of the subject keys (`nodeName`, `materialName` or `tagName`)
65
+ */
66
+ export type ParameterValues = (ParameterSubject & {
67
+ parameterName: ParameterName;
68
+ value: ParameterValue;
69
+ })[];
70
+
71
+ /**
72
+ * Definition of callback function for parameter change
73
+ */
74
+ export type ParameterObserver = (payload: ParameterObserverPayload) => Promise<void>;
75
+
76
+ /**
77
+ * Payload of parameter observer.\
78
+ * Contains current data of parameter entry, which can be usefull for implementing the dedicated observer
79
+ */
80
+ export type ParameterObserverPayload = {
81
+ subject: ParameterSubject;
82
+ nodes: TransformNode[];
83
+ materials: Material[];
84
+ newValue: ParameterValue;
85
+ oldValue: ParameterValue | undefined;
86
+ };
87
+
88
+ // internal type which holds the data of a certain parameter entry
89
+ type ParameterEntry = {
90
+ subject: ParameterSubject;
91
+ parameterName: ParameterName;
92
+ value: ParameterValue;
93
+ oldValue: ParameterValue | undefined;
94
+ };
95
+
96
+ export class ParameterManager {
97
+ /**
98
+ * Parses and converts input to a boolean value, valid values are:
99
+ * - true / false
100
+ * - 1 / 0
101
+ */
102
+ public static parseBoolean(value: ParameterValue): boolean {
103
+ if (value.toString() === 'true' || value.toString() === '1') {
104
+ return true;
105
+ } else if (value.toString() === 'false' || value.toString() === '0') {
106
+ return false;
107
+ }
108
+ throw new ViewerError({
109
+ id: ViewerErrorIds.InvalidParameterValue,
110
+ message: `Unable to parse "${value}" to a boolean`,
111
+ });
112
+ }
113
+
114
+ /**
115
+ * Parses and converts input to a number value
116
+ */
117
+ public static parseNumber(value: ParameterValue): number {
118
+ return parseFloat(value.toString());
119
+ }
120
+
121
+ /**
122
+ * Parses and converts input to a string value
123
+ */
124
+ public static parseString(value: ParameterValue): string {
125
+ return value.toString();
126
+ }
127
+
128
+ // TODO WTT: enable setting Vector3 on the input directly
129
+ /**
130
+ * Parses a string of format "(x,y,z)"" to a "Vector3".
131
+ */
132
+ public static parseVector(value: ParameterValue): Vector3 {
133
+ if (!isString(value)) {
134
+ throw new ViewerError({
135
+ id: ViewerErrorIds.InvalidParameterValue,
136
+ message: `Unable to parse "${value}" to a vector: not a string`,
137
+ });
138
+ }
139
+ let cleanedValue = value.split(' ').join('');
140
+ if (cleanedValue.startsWith('(') && cleanedValue.endsWith(')')) {
141
+ cleanedValue = cleanedValue.substring(1, cleanedValue.length - 1);
142
+ const [x, y, z] = cleanedValue.split(',').map(value => parseFloat(value));
143
+ return new Vector3(x, y, z);
144
+ } else {
145
+ throw new ViewerError({
146
+ id: ViewerErrorIds.InvalidParameterValue,
147
+ message: `Unable to parse "${value}" to a vector: expected "(x,y,z)"`,
148
+ });
149
+ }
150
+ }
151
+
152
+ // TODO WTT: enable setting Vector3 on the input directly (maybe quaternion as well)
153
+ /**
154
+ * Parses a string of format `'(x,y,z)'` with angular degrees to a `Vector3` with rotation information in radians.
155
+ */
156
+ public static parseRotation(value: ParameterValue): Vector3 {
157
+ const rotation = ParameterManager.parseVector(value);
158
+ const deg2rad = (deg: number): number => {
159
+ return (deg * Math.PI) / 180;
160
+ };
161
+ return rotation.set(deg2rad(rotation.x), deg2rad(rotation.y), deg2rad(rotation.z));
162
+ }
163
+
164
+ // TODO WTT: enable setting Color3 on the input directly
165
+ /**
166
+ * Parses a string of format `'#rrggbb'` or `'(r,g,b)'` to a `Color3`.
167
+ */
168
+ public static parseColor(value: ParameterValue): Color3 {
169
+ const cleanedValue = value.toString().split(' ').join('');
170
+ if (cleanedValue.startsWith('#')) {
171
+ return Color3.FromHexString(value.toString());
172
+ }
173
+ if (cleanedValue.startsWith('(') && cleanedValue.endsWith(')')) {
174
+ const rgb = cleanedValue.substring(1, cleanedValue.length - 1);
175
+ const [r, g, b] = rgb.split(',').map(value => parseFloat(value));
176
+ return Color3.FromInts(r, g, b);
177
+ }
178
+ const humanReadable = capitalize(cleanedValue);
179
+
180
+ if (Object.prototype.hasOwnProperty.call(Color3, humanReadable)) {
181
+ return (Color3 as any)[humanReadable]();
182
+ }
183
+ throw new ViewerError({
184
+ id: ViewerErrorIds.InvalidParameterValue,
185
+ message: `Unable to parse "${value}" to a color: expected "#rrggbb", "(r,g,b)" or any human readable (e.g. Red) property implemented in Color3`,
186
+ });
187
+ }
188
+
189
+ protected _parameterEntries: ParameterEntry[] = [];
190
+ protected _parameterObserver: { [parameterName: ParameterName]: ParameterObserver } = {};
191
+
192
+ public constructor(protected viewer: Viewer) {
193
+ this._addBuiltInParameterObservers();
194
+ }
195
+
196
+ /**
197
+ * Set parameter value for a certain node and calls the corresponding observer if the value has changed
198
+ *
199
+ * @returns "true" if parameter value has changed
200
+ */
201
+ public async setNodeParameterValue(
202
+ nodeName: string,
203
+ parameterName: ParameterName,
204
+ value: ParameterValue
205
+ ): Promise<boolean> {
206
+ const valueChanged = this._addParameterValue({ nodeName }, parameterName, value);
207
+ if (valueChanged) {
208
+ await this._applyParameterValue({ nodeName }, parameterName);
209
+ }
210
+
211
+ return valueChanged;
212
+ }
213
+
214
+ /**
215
+ * Set parameter value for a certain material and calls the corresponding observer if the value has changed
216
+ *
217
+ * @returns "true" if parameter value has changed
218
+ */
219
+ public async setMaterialParameterValue(
220
+ materialName: string,
221
+ parameterName: ParameterName,
222
+ value: ParameterValue
223
+ ): Promise<boolean> {
224
+ const valueChanged = this._addParameterValue({ materialName }, parameterName, value);
225
+ if (valueChanged) {
226
+ await this._applyParameterValue({ materialName }, parameterName);
227
+ }
228
+
229
+ return valueChanged;
230
+ }
231
+
232
+ /**
233
+ * Set parameter value for a certain tag and calls the corresponding observer if the value has changed.\
234
+ * Setting a parameter value on a tag can affect multiple nodes and tags, depending which of these objects contains
235
+ * the desired tag.
236
+ *
237
+ * @returns "true" if parameter value has changed
238
+ */
239
+ public async setTagParameterValue(
240
+ tagName: string,
241
+ parameterName: ParameterName,
242
+ value: ParameterValue
243
+ ): Promise<boolean> {
244
+ const valueChanged = this._addParameterValue({ tagName }, parameterName, value);
245
+ if (valueChanged) {
246
+ await this._applyParameterValue({ tagName }, parameterName);
247
+ }
248
+
249
+ return valueChanged;
250
+ }
251
+
252
+ /**
253
+ * Set multiple parameter values simultaniously.\
254
+ * Tag parameters are applied before node and material parameters, node and materials are more specific and should
255
+ * have priority.
256
+ *
257
+ * @returns Array of parameters, which have changed values
258
+ */
259
+ public async setParameterValues(values: ParameterValues): Promise<ParameterValues> {
260
+ const parameterEntries = values.map<ParameterEntry>(valueEntry => {
261
+ const subject: ParameterSubject = isNodeParameterSubject(valueEntry)
262
+ ? { nodeName: valueEntry.nodeName }
263
+ : isMaterialParameterSubject(valueEntry)
264
+ ? { materialName: valueEntry.materialName }
265
+ : { tagName: valueEntry.tagName };
266
+
267
+ return { subject, parameterName: valueEntry.parameterName, value: valueEntry.value, oldValue: undefined };
268
+ });
269
+
270
+ const changedParameterEntries = parameterEntries.filter(paramEntry =>
271
+ this._addParameterValue(paramEntry.subject, paramEntry.parameterName, paramEntry.value)
272
+ );
273
+
274
+ await this._applyParameterValues(changedParameterEntries);
275
+
276
+ // convert back to original typing
277
+ const changedParameterValues: ParameterValues = changedParameterEntries.map(paramEntry => {
278
+ return { ...paramEntry.subject, parameterName: paramEntry.parameterName, value: paramEntry.value };
279
+ });
280
+
281
+ return changedParameterValues;
282
+ }
283
+
284
+ /**
285
+ * @returns desired parameter value or "undefined" if parameter entry is not available.
286
+ */
287
+ public getParameterValue(subject: ParameterSubject, parameterName: ParameterName): ParameterValue | undefined {
288
+ const entry = this._getEntry(subject, parameterName);
289
+
290
+ return entry?.value;
291
+ }
292
+
293
+ /**
294
+ * Define observer callback for certain parameter.\
295
+ * There can only be one observer for a certain parameter name.\
296
+ * Parameter observers can not be overwritten once they are defined, this also includes system observers for
297
+ * {@link BuiltInParameter}.
298
+ */
299
+ public setParameterObserver(parameterName: ParameterName, observer: ParameterObserver): void {
300
+ if (this._parameterObserver[parameterName]) {
301
+ console.warn(`Observer for parameter "${parameterName}" already set`);
302
+ return;
303
+ }
304
+
305
+ this._parameterObserver[parameterName] = observer;
306
+ }
307
+
308
+ /**
309
+ * Print all parameter entries in table format into the console
310
+ */
311
+ public printAllParameters(): void {
312
+ const printable = this._parameterEntries.map(entry => ({
313
+ ...entry,
314
+ subject: JSON.stringify(entry.subject),
315
+ }));
316
+
317
+ console.table(printable);
318
+ }
319
+
320
+ /**
321
+ * Applies all existing parameter entries to a certain "model", as defined in the {@link ModelManager}.\
322
+ * This can be usefull when updating the model before showing it in the scene.
323
+ *
324
+ * @internal
325
+ */
326
+ public async applyAllParameterValuesToModel(model: AbstractScene): Promise<void> {
327
+ const parameterEntriesToApply = this._parameterEntries;
328
+
329
+ await this._applyParameterValues(parameterEntriesToApply, model);
330
+ }
331
+
332
+ /**
333
+ * Applies all parameter values which are targeting a material.\
334
+ * This can be usefull when updating a material definition before creating it.
335
+ *
336
+ * @internal
337
+ */
338
+ public async applyParameterValuesToMaterial(material: Material): Promise<void> {
339
+ const tags = getTags(material);
340
+
341
+ const parameterEntriesToApply: ParameterEntry[] = [];
342
+ tags.forEach(tagName => {
343
+ const tagParamEntries = this._getEntriesOfSubject({ tagName });
344
+ parameterEntriesToApply.push(...tagParamEntries);
345
+ });
346
+
347
+ const materialParamEntries = this._getEntriesOfSubject({ materialName: material.id });
348
+ parameterEntriesToApply.push(...materialParamEntries);
349
+
350
+ await this._applyParameterValues(parameterEntriesToApply);
351
+ }
352
+
353
+ /**
354
+ * @returns Desired parameter value of a certain node.
355
+ * Tags are considered as well but have lower priority than node parameters, as these are more specific.
356
+ *
357
+ * @internal
358
+ */
359
+ public getParameterValueOfNode(node: TransformNode, parameterName: string): ParameterValue | undefined {
360
+ const nodeParamValue = this.getParameterValue({ nodeName: node.name }, parameterName);
361
+ if (nodeParamValue !== undefined) {
362
+ return nodeParamValue;
363
+ }
364
+
365
+ const tags = getTags(node);
366
+ const tagParamValue = tags.reduce<ParameterValue | undefined>((accValue, curTag) => {
367
+ // NOTE: it is possible that values are available for multiple tags
368
+ // in this case the resulting parameter value is quite "random" as the last tag in the tag string of the node has
369
+ // priority
370
+ const tagParamValue = this.getParameterValue({ tagName: curTag }, parameterName);
371
+ return accValue ?? tagParamValue;
372
+ }, undefined);
373
+
374
+ return tagParamValue;
375
+ }
376
+
377
+ /**
378
+ * @returns Desired parameter value of a certain material.
379
+ * Tags are considered as well but have lower priority than material parameters, as these are more specific.
380
+ * Unused ATM, but added for consistency as counter part for {@link getParameterValueOfNode}
381
+ *
382
+ * @internal
383
+ */
384
+ public getParameterValueOfMaterial(material: Material, parameterName: string): ParameterValue | undefined {
385
+ const materialParamValue = this.getParameterValue({ materialName: material.name }, parameterName);
386
+ if (materialParamValue !== undefined) {
387
+ return materialParamValue;
388
+ }
389
+
390
+ const tags = getTags(material);
391
+ const tagParamValue = tags.reduce<ParameterValue | undefined>((accValue, curTag) => {
392
+ const tagParamValue = this.getParameterValue({ tagName: curTag }, parameterName);
393
+ return accValue ?? tagParamValue;
394
+ }, undefined);
395
+
396
+ return tagParamValue;
397
+ }
398
+
399
+ /**
400
+ * Parameter observer implementation of default parameters
401
+ */
402
+ protected _addBuiltInParameterObservers(): void {
403
+ this.setParameterObserver(BuiltInParameter.Visible, async ({ nodes, newValue }) => {
404
+ const visible = ParameterManager.parseBoolean(newValue);
405
+
406
+ for (const node of nodes) {
407
+ if (visible) {
408
+ const deferredMaterial = getInternalMetadataValue(node, 'deferredMaterial');
409
+ if (deferredMaterial) {
410
+ await this.viewer.materialManager.setMaterialOnMesh(deferredMaterial as string, node as AbstractMesh);
411
+ }
412
+
413
+ node.setEnabled(true);
414
+ } else {
415
+ node.setEnabled(false);
416
+ }
417
+ }
418
+ });
419
+ this.setParameterObserver(BuiltInParameter.Material, async ({ nodes, newValue }) => {
420
+ const material = ParameterManager.parseString(newValue);
421
+
422
+ for (const node of nodes) {
423
+ // NOTE: don't use node.isEnabled() as visibility observer is probably called in same cycle but later
424
+ // however the parameter value is already correct at this stage
425
+ const rawVisibleValue = this.getParameterValueOfNode(node, BuiltInParameter.Visible);
426
+ const visible =
427
+ rawVisibleValue !== undefined ? ParameterManager.parseBoolean(rawVisibleValue) : node.isEnabled();
428
+ if (visible) {
429
+ // TODO WTT: check mesh type and throw error if it doesn't fit
430
+ // think of creating a framework with type guards (isMesh, canHaveMaterial, ...) around this
431
+ await this.viewer.materialManager.setMaterialOnMesh(material, node as AbstractMesh);
432
+ } else {
433
+ setInternalMetadataValue(node, 'deferredMaterial', material);
434
+ }
435
+ }
436
+ });
437
+ this.setParameterObserver(BuiltInParameter.Position, async ({ nodes, newValue }) => {
438
+ const position = ParameterManager.parseVector(newValue);
439
+
440
+ for (const node of nodes) {
441
+ node.position = position;
442
+ }
443
+ });
444
+ this.setParameterObserver(BuiltInParameter.Rotation, async ({ nodes, newValue }) => {
445
+ const rotation = ParameterManager.parseRotation(newValue);
446
+
447
+ for (const node of nodes) {
448
+ node.rotation = rotation;
449
+ }
450
+ });
451
+ this.setParameterObserver(BuiltInParameter.Scaling, async ({ nodes, newValue }) => {
452
+ const scaling = ParameterManager.parseVector(newValue);
453
+
454
+ for (const node of nodes) {
455
+ node.scaling = scaling;
456
+ }
457
+ });
458
+ this.setParameterObserver(BuiltInParameter.Color, async ({ materials, newValue }) => {
459
+ const color = ParameterManager.parseColor(newValue);
460
+
461
+ for (const material of materials) {
462
+ const materialCls = material.getClassName();
463
+ switch (materialCls) {
464
+ case 'PBRMaterial':
465
+ (material as PBRMaterial).albedoColor = color.toLinearSpace();
466
+ break;
467
+ case 'StandardMaterial':
468
+ (material as StandardMaterial).diffuseColor = color;
469
+ break;
470
+ default:
471
+ throw new Error(`Setting color for material of instance "${materialCls}" not implemented`);
472
+ }
473
+ }
474
+ });
475
+ this.setParameterObserver(BuiltInParameter.Roughness, async ({ materials, newValue }) => {
476
+ const roughness = ParameterManager.parseNumber(newValue);
477
+
478
+ for (const material of materials) {
479
+ const materialCls = material.getClassName();
480
+ switch (materialCls) {
481
+ case 'PBRMaterial':
482
+ (material as PBRMaterial).roughness = roughness;
483
+ break;
484
+ case 'StandardMaterial':
485
+ (material as StandardMaterial).roughness = roughness;
486
+ break;
487
+ default:
488
+ throw new Error(`Setting rougness for material of instance "${materialCls}" not implemented`);
489
+ }
490
+ }
491
+ });
492
+ this.setParameterObserver(BuiltInParameter.Metallic, async ({ materials, newValue }) => {
493
+ const metallic = ParameterManager.parseNumber(newValue);
494
+
495
+ for (const material of materials) {
496
+ const materialCls = material.getClassName();
497
+ switch (materialCls) {
498
+ case 'PBRMaterial':
499
+ (material as PBRMaterial).metallic = metallic;
500
+ break;
501
+ default:
502
+ throw new Error(`Setting metallic for material of instance "${materialCls}" not implemented`);
503
+ }
504
+ }
505
+ });
506
+ this.setParameterObserver(BuiltInParameter.Paintable, async ({ newValue, materials }) => {
507
+ paintableParameterObserver(newValue, materials, this.viewer.scene);
508
+ });
509
+ }
510
+
511
+ /**
512
+ * Change parameter value in array of existing parameter entries or create a new entry
513
+ *
514
+ * @returns "true" if parameter has changed or wasn't available before
515
+ */
516
+ protected _addParameterValue(
517
+ subject: ParameterSubject,
518
+ parameterName: ParameterName,
519
+ value: ParameterValue
520
+ ): boolean {
521
+ const existingEntry = this._getEntry(subject, parameterName);
522
+
523
+ if (existingEntry?.value === value) {
524
+ return false;
525
+ }
526
+
527
+ if (existingEntry) {
528
+ existingEntry.oldValue = existingEntry.value;
529
+ existingEntry.value = value;
530
+ } else {
531
+ this._parameterEntries.push({ subject, parameterName, value, oldValue: undefined });
532
+ }
533
+
534
+ return true;
535
+ }
536
+
537
+ /**
538
+ * Call parameter observer of desired parameter which usually apply the new values to the scene.
539
+ *
540
+ * @param assetContainer Asset container in which to look for the paramter entries subjects (e.g. the nodes and
541
+ * materials to which the parameter values should be applied to).\
542
+ * Defaults to `viewer.scene`.
543
+ */
544
+ protected async _applyParameterValues(
545
+ parameterEntries: ParameterEntry[],
546
+ assetContainer?: AbstractScene
547
+ ): Promise<void> {
548
+ const tagParamEntries = parameterEntries.filter(entry => isTagParameterSubject(entry.subject));
549
+ const nonTagParamEntries = parameterEntries.filter(entry => !isTagParameterSubject(entry.subject));
550
+
551
+ for (const entry of tagParamEntries) {
552
+ await this._applyParameterValue(entry.subject, entry.parameterName, assetContainer);
553
+ }
554
+ for (const entry of nonTagParamEntries) {
555
+ await this._applyParameterValue(entry.subject, entry.parameterName, assetContainer);
556
+ }
557
+ }
558
+
559
+ /**
560
+ * Call parameter observer of desired parameter
561
+ *
562
+ * @param assetContainer Optionally add an "asset container", which actually represents a model in the
563
+ * {@link ModelManager}. Viewer scene is used if left empty.
564
+ */
565
+ protected async _applyParameterValue(
566
+ subject: ParameterSubject,
567
+ parameterName: ParameterName,
568
+ assetContainer?: AbstractScene
569
+ ): Promise<void> {
570
+ const parameterEntry = this._getEntry(subject, parameterName);
571
+ const observer = this._parameterObserver[parameterName];
572
+ if (!parameterEntry || !observer) {
573
+ return;
574
+ }
575
+
576
+ const nodes = this._getAffectedNodes(subject, assetContainer);
577
+ const materials = this._getAffectedMaterials(subject, assetContainer);
578
+
579
+ await observer({
580
+ subject,
581
+ newValue: parameterEntry.value,
582
+ oldValue: parameterEntry.oldValue,
583
+ nodes,
584
+ materials,
585
+ });
586
+ }
587
+
588
+ protected _getAffectedMaterials(subject: ParameterSubject, assetContainer?: AbstractScene): Material[] {
589
+ assetContainer = assetContainer ?? this.viewer.scene;
590
+ let materials: Material[] = [];
591
+
592
+ // materials have priority over tags
593
+ if (isMaterialParameterSubject(subject)) {
594
+ materials = assetContainer.materials.filter(material => material.name === subject.materialName);
595
+ if (materials.length > 1) {
596
+ console.warn(`Multiple materials for material name "${subject.materialName}" have been found`);
597
+ }
598
+ } else if (isTagParameterSubject(subject)) {
599
+ materials = assetContainer.materials.filter(material => hasTag(material, subject.tagName));
600
+ }
601
+
602
+ return materials;
603
+ }
604
+
605
+ protected _getAffectedNodes(subject: ParameterSubject, assetContainer?: AbstractScene): TransformNode[] {
606
+ assetContainer = assetContainer ?? this.viewer.scene;
607
+ const allNodes = [...assetContainer.meshes, ...assetContainer.transformNodes];
608
+ let nodes: TransformNode[] = [];
609
+
610
+ // nodes have priority over tags
611
+ if (isNodeParameterSubject(subject)) {
612
+ nodes = allNodes.filter(node => node.name === subject.nodeName);
613
+ if (nodes.length > 1) {
614
+ console.warn(`Multiple nodes for node name "${subject.nodeName}" have been found`);
615
+ }
616
+ } else if (isTagParameterSubject(subject)) {
617
+ nodes = allNodes.filter(node => hasTag(node, subject.tagName));
618
+ }
619
+
620
+ return nodes;
621
+ }
622
+
623
+ protected _getEntry(subject: ParameterSubject, parameterName: ParameterName): ParameterEntry | undefined {
624
+ const entriesOfSubject = this._getEntriesOfSubject(subject);
625
+ const entry = entriesOfSubject.find(entry => entry.parameterName === parameterName);
626
+
627
+ return entry;
628
+ }
629
+
630
+ protected _getEntriesOfSubject(subject: ParameterSubject): ParameterEntry[] {
631
+ const entries = this._parameterEntries.filter(entry => {
632
+ const nodeNameMatches =
633
+ isNodeParameterSubject(entry.subject) &&
634
+ isNodeParameterSubject(subject) &&
635
+ entry.subject.nodeName === subject.nodeName;
636
+
637
+ const materialNameMatches =
638
+ isMaterialParameterSubject(entry.subject) &&
639
+ isMaterialParameterSubject(subject) &&
640
+ entry.subject.materialName === subject.materialName;
641
+
642
+ const tagNameMatches =
643
+ isTagParameterSubject(entry.subject) &&
644
+ isTagParameterSubject(subject) &&
645
+ entry.subject.tagName === subject.tagName;
646
+
647
+ return nodeNameMatches || materialNameMatches || tagNameMatches;
648
+ });
649
+
650
+ return entries;
651
+ }
652
+ }