@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.
- package/dist/lib-cjs/buildinfo.json +1 -1
- package/dist/lib-cjs/commonjs.tsconfig.tsbuildinfo +1 -1
- package/dist/lib-cjs/index.d.ts +63 -63
- package/dist/lib-cjs/index.js +81 -114
- package/dist/lib-cjs/index.js.map +1 -1
- package/dist/lib-cjs/internal/cbnCustomBabylonLoaderPlugin.d.ts +10 -0
- package/dist/lib-cjs/internal/cbnCustomBabylonLoaderPlugin.js +124 -0
- package/dist/lib-cjs/internal/cbnCustomBabylonLoaderPlugin.js.map +1 -0
- package/dist/lib-cjs/internal/cloningHelper.d.ts +19 -0
- package/dist/lib-cjs/internal/cloningHelper.js +165 -0
- package/dist/lib-cjs/internal/cloningHelper.js.map +1 -0
- package/dist/lib-cjs/internal/deviceHelper.d.ts +9 -0
- package/dist/lib-cjs/{api/util → internal}/deviceHelper.js +6 -10
- package/dist/lib-cjs/internal/deviceHelper.js.map +1 -0
- package/dist/lib-cjs/internal/geometryHelper.d.ts +21 -0
- package/dist/lib-cjs/{api/util → internal}/geometryHelper.js +57 -24
- package/dist/lib-cjs/internal/geometryHelper.js.map +1 -0
- package/dist/lib-cjs/internal/metadataHelper.d.ts +26 -0
- package/dist/lib-cjs/internal/metadataHelper.js +51 -0
- package/dist/lib-cjs/internal/metadataHelper.js.map +1 -0
- package/dist/lib-cjs/internal/paintableHelper.d.ts +40 -0
- package/dist/lib-cjs/internal/paintableHelper.js +287 -0
- package/dist/lib-cjs/internal/paintableHelper.js.map +1 -0
- package/dist/lib-cjs/internal/tagsHelper.d.ts +12 -0
- package/dist/lib-cjs/internal/tagsHelper.js +38 -0
- package/dist/lib-cjs/internal/tagsHelper.js.map +1 -0
- package/dist/lib-cjs/manager/cameraManager.d.ts +51 -0
- package/dist/lib-cjs/manager/cameraManager.js +154 -0
- package/dist/lib-cjs/manager/cameraManager.js.map +1 -0
- package/dist/lib-cjs/manager/debugManager.d.ts +60 -0
- package/dist/lib-cjs/manager/debugManager.js +218 -0
- package/dist/lib-cjs/manager/debugManager.js.map +1 -0
- package/dist/lib-cjs/manager/eventManager.d.ts +52 -0
- package/dist/lib-cjs/manager/eventManager.js +72 -0
- package/dist/lib-cjs/manager/eventManager.js.map +1 -0
- package/dist/lib-cjs/{api/manager → manager}/gltfExportManager.d.ts +29 -34
- package/dist/lib-cjs/{api/manager → manager}/gltfExportManager.js +99 -120
- package/dist/lib-cjs/manager/gltfExportManager.js.map +1 -0
- package/dist/lib-cjs/manager/materialManager.d.ts +35 -0
- package/dist/lib-cjs/manager/materialManager.js +126 -0
- package/dist/lib-cjs/manager/materialManager.js.map +1 -0
- package/dist/lib-cjs/manager/modelManager.d.ts +145 -0
- package/dist/lib-cjs/manager/modelManager.js +381 -0
- package/dist/lib-cjs/manager/modelManager.js.map +1 -0
- package/dist/lib-cjs/manager/parameterManager.d.ts +210 -0
- package/dist/lib-cjs/manager/parameterManager.js +515 -0
- package/dist/lib-cjs/manager/parameterManager.js.map +1 -0
- package/dist/lib-cjs/manager/sceneManager.d.ts +45 -0
- package/dist/lib-cjs/manager/sceneManager.js +65 -0
- package/dist/lib-cjs/manager/sceneManager.js.map +1 -0
- package/dist/lib-cjs/manager/screenshotManager.d.ts +36 -0
- package/dist/lib-cjs/manager/screenshotManager.js +40 -0
- package/dist/lib-cjs/manager/screenshotManager.js.map +1 -0
- package/dist/lib-cjs/manager/textureManager.d.ts +12 -0
- package/dist/lib-cjs/manager/textureManager.js +44 -0
- package/dist/lib-cjs/manager/textureManager.js.map +1 -0
- package/dist/lib-cjs/viewer.d.ts +117 -0
- package/dist/lib-cjs/viewer.js +222 -0
- package/dist/lib-cjs/viewer.js.map +1 -0
- package/dist/lib-cjs/{api/classes/viewerError.d.ts → viewerError.d.ts} +6 -1
- package/dist/lib-cjs/{api/classes/viewerError.js → viewerError.js} +6 -1
- package/dist/lib-cjs/viewerError.js.map +1 -0
- package/package.json +10 -11
- package/src/dev.ts +14 -37
- package/src/{types.d.ts → globalTypes.d.ts} +8 -18
- package/src/index.ts +79 -113
- package/src/internal/cbnCustomBabylonLoaderPlugin.ts +149 -0
- package/src/internal/cloningHelper.ts +225 -0
- package/src/internal/deviceHelper.ts +25 -0
- package/src/{api/util → internal}/geometryHelper.ts +63 -24
- package/src/internal/metadataHelper.ts +63 -0
- package/src/internal/paintableHelper.ts +310 -0
- package/src/internal/tagsHelper.ts +41 -0
- package/src/manager/cameraManager.ts +236 -0
- package/src/manager/debugManager.ts +245 -0
- package/src/manager/eventManager.ts +72 -0
- package/src/{api/manager → manager}/gltfExportManager.ts +132 -125
- package/src/manager/materialManager.ts +135 -0
- package/src/manager/modelManager.ts +456 -0
- package/src/manager/parameterManager.ts +652 -0
- package/src/manager/sceneManager.ts +101 -0
- package/src/manager/screenshotManager.ts +59 -0
- package/src/manager/textureManager.ts +32 -0
- package/src/viewer.ts +296 -0
- package/src/{api/classes/viewerError.ts → viewerError.ts} +6 -1
- package/dist/lib-cjs/api/classes/animationInterface.d.ts +0 -8
- package/dist/lib-cjs/api/classes/animationInterface.js +0 -3
- package/dist/lib-cjs/api/classes/animationInterface.js.map +0 -1
- package/dist/lib-cjs/api/classes/dottedPath.d.ts +0 -79
- package/dist/lib-cjs/api/classes/dottedPath.js +0 -167
- package/dist/lib-cjs/api/classes/dottedPath.js.map +0 -1
- package/dist/lib-cjs/api/classes/element.d.ts +0 -153
- package/dist/lib-cjs/api/classes/element.js +0 -703
- package/dist/lib-cjs/api/classes/element.js.map +0 -1
- package/dist/lib-cjs/api/classes/event.d.ts +0 -401
- package/dist/lib-cjs/api/classes/event.js +0 -425
- package/dist/lib-cjs/api/classes/event.js.map +0 -1
- package/dist/lib-cjs/api/classes/eventBroadcaster.d.ts +0 -26
- package/dist/lib-cjs/api/classes/eventBroadcaster.js +0 -50
- package/dist/lib-cjs/api/classes/eventBroadcaster.js.map +0 -1
- package/dist/lib-cjs/api/classes/fuzzyMap.d.ts +0 -7
- package/dist/lib-cjs/api/classes/fuzzyMap.js +0 -22
- package/dist/lib-cjs/api/classes/fuzzyMap.js.map +0 -1
- package/dist/lib-cjs/api/classes/parameter.d.ts +0 -410
- package/dist/lib-cjs/api/classes/parameter.js +0 -643
- package/dist/lib-cjs/api/classes/parameter.js.map +0 -1
- package/dist/lib-cjs/api/classes/parameterObservable.d.ts +0 -36
- package/dist/lib-cjs/api/classes/parameterObservable.js +0 -73
- package/dist/lib-cjs/api/classes/parameterObservable.js.map +0 -1
- package/dist/lib-cjs/api/classes/parameterizable.d.ts +0 -15
- package/dist/lib-cjs/api/classes/parameterizable.js +0 -103
- package/dist/lib-cjs/api/classes/parameterizable.js.map +0 -1
- package/dist/lib-cjs/api/classes/placementAnimation.d.ts +0 -45
- package/dist/lib-cjs/api/classes/placementAnimation.js +0 -177
- package/dist/lib-cjs/api/classes/placementAnimation.js.map +0 -1
- package/dist/lib-cjs/api/classes/variant.d.ts +0 -261
- package/dist/lib-cjs/api/classes/variant.js +0 -873
- package/dist/lib-cjs/api/classes/variant.js.map +0 -1
- package/dist/lib-cjs/api/classes/variantInstance.d.ts +0 -53
- package/dist/lib-cjs/api/classes/variantInstance.js +0 -126
- package/dist/lib-cjs/api/classes/variantInstance.js.map +0 -1
- package/dist/lib-cjs/api/classes/variantParameterizable.d.ts +0 -17
- package/dist/lib-cjs/api/classes/variantParameterizable.js +0 -87
- package/dist/lib-cjs/api/classes/variantParameterizable.js.map +0 -1
- package/dist/lib-cjs/api/classes/viewer.d.ts +0 -215
- package/dist/lib-cjs/api/classes/viewer.js +0 -709
- package/dist/lib-cjs/api/classes/viewer.js.map +0 -1
- package/dist/lib-cjs/api/classes/viewerError.js.map +0 -1
- package/dist/lib-cjs/api/classes/viewerLight.d.ts +0 -66
- package/dist/lib-cjs/api/classes/viewerLight.js +0 -345
- package/dist/lib-cjs/api/classes/viewerLight.js.map +0 -1
- package/dist/lib-cjs/api/internal/lensRendering.d.ts +0 -8
- package/dist/lib-cjs/api/internal/lensRendering.js +0 -12
- package/dist/lib-cjs/api/internal/lensRendering.js.map +0 -1
- package/dist/lib-cjs/api/internal/sceneSetup.d.ts +0 -13
- package/dist/lib-cjs/api/internal/sceneSetup.js +0 -228
- package/dist/lib-cjs/api/internal/sceneSetup.js.map +0 -1
- package/dist/lib-cjs/api/manager/animationManager.d.ts +0 -30
- package/dist/lib-cjs/api/manager/animationManager.js +0 -127
- package/dist/lib-cjs/api/manager/animationManager.js.map +0 -1
- package/dist/lib-cjs/api/manager/gltfExportManager.js.map +0 -1
- package/dist/lib-cjs/api/manager/sceneManager.d.ts +0 -33
- package/dist/lib-cjs/api/manager/sceneManager.js +0 -129
- package/dist/lib-cjs/api/manager/sceneManager.js.map +0 -1
- package/dist/lib-cjs/api/manager/tagManager.d.ts +0 -118
- package/dist/lib-cjs/api/manager/tagManager.js +0 -531
- package/dist/lib-cjs/api/manager/tagManager.js.map +0 -1
- package/dist/lib-cjs/api/manager/textureLoadManager.d.ts +0 -22
- package/dist/lib-cjs/api/manager/textureLoadManager.js +0 -108
- package/dist/lib-cjs/api/manager/textureLoadManager.js.map +0 -1
- package/dist/lib-cjs/api/manager/variantInstanceManager.d.ts +0 -106
- package/dist/lib-cjs/api/manager/variantInstanceManager.js +0 -291
- package/dist/lib-cjs/api/manager/variantInstanceManager.js.map +0 -1
- package/dist/lib-cjs/api/store/specStorage.d.ts +0 -32
- package/dist/lib-cjs/api/store/specStorage.js +0 -66
- package/dist/lib-cjs/api/store/specStorage.js.map +0 -1
- package/dist/lib-cjs/api/util/babylonHelper.d.ts +0 -238
- package/dist/lib-cjs/api/util/babylonHelper.js +0 -826
- package/dist/lib-cjs/api/util/babylonHelper.js.map +0 -1
- package/dist/lib-cjs/api/util/debugHelper.d.ts +0 -9
- package/dist/lib-cjs/api/util/debugHelper.js +0 -94
- package/dist/lib-cjs/api/util/debugHelper.js.map +0 -1
- package/dist/lib-cjs/api/util/deviceHelper.d.ts +0 -9
- package/dist/lib-cjs/api/util/deviceHelper.js.map +0 -1
- package/dist/lib-cjs/api/util/geometryHelper.d.ts +0 -17
- package/dist/lib-cjs/api/util/geometryHelper.js.map +0 -1
- package/dist/lib-cjs/api/util/globalTypes.d.ts +0 -490
- package/dist/lib-cjs/api/util/globalTypes.js +0 -2
- package/dist/lib-cjs/api/util/globalTypes.js.map +0 -1
- package/dist/lib-cjs/api/util/resourceHelper.d.ts +0 -58
- package/dist/lib-cjs/api/util/resourceHelper.js +0 -215
- package/dist/lib-cjs/api/util/resourceHelper.js.map +0 -1
- package/dist/lib-cjs/api/util/sceneLoaderHelper.d.ts +0 -58
- package/dist/lib-cjs/api/util/sceneLoaderHelper.js +0 -229
- package/dist/lib-cjs/api/util/sceneLoaderHelper.js.map +0 -1
- package/dist/lib-cjs/api/util/stringHelper.d.ts +0 -13
- package/dist/lib-cjs/api/util/stringHelper.js +0 -33
- package/dist/lib-cjs/api/util/stringHelper.js.map +0 -1
- package/dist/lib-cjs/api/util/structureHelper.d.ts +0 -9
- package/dist/lib-cjs/api/util/structureHelper.js +0 -58
- package/dist/lib-cjs/api/util/structureHelper.js.map +0 -1
- package/src/api/classes/animationInterface.ts +0 -10
- package/src/api/classes/dottedPath.ts +0 -181
- package/src/api/classes/element.ts +0 -766
- package/src/api/classes/event.ts +0 -457
- package/src/api/classes/eventBroadcaster.ts +0 -52
- package/src/api/classes/fuzzyMap.ts +0 -21
- package/src/api/classes/parameter.ts +0 -686
- package/src/api/classes/parameterObservable.ts +0 -73
- package/src/api/classes/parameterizable.ts +0 -87
- package/src/api/classes/placementAnimation.ts +0 -162
- package/src/api/classes/variant.ts +0 -965
- package/src/api/classes/variantInstance.ts +0 -123
- package/src/api/classes/variantParameterizable.ts +0 -83
- package/src/api/classes/viewer.ts +0 -751
- package/src/api/classes/viewerLight.ts +0 -335
- package/src/api/internal/debugViewer.ts +0 -90
- package/src/api/internal/lensRendering.ts +0 -9
- package/src/api/internal/sceneSetup.ts +0 -208
- package/src/api/manager/animationManager.ts +0 -143
- package/src/api/manager/sceneManager.ts +0 -134
- package/src/api/manager/tagManager.ts +0 -572
- package/src/api/manager/textureLoadManager.ts +0 -107
- package/src/api/manager/variantInstanceManager.ts +0 -306
- package/src/api/store/specStorage.ts +0 -68
- package/src/api/util/babylonHelper.ts +0 -915
- package/src/api/util/debugHelper.ts +0 -121
- package/src/api/util/deviceHelper.ts +0 -31
- package/src/api/util/globalTypes.ts +0 -566
- package/src/api/util/resourceHelper.ts +0 -201
- package/src/api/util/sceneLoaderHelper.ts +0 -247
- package/src/api/util/stringHelper.ts +0 -30
- 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
|
+
}
|