@combeenation/3d-viewer 4.0.0-beta4 → 4.0.1-alpha1

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 (106) hide show
  1. package/README.md +114 -112
  2. package/dist/lib-cjs/api/classes/animationInterface.d.ts +8 -8
  3. package/dist/lib-cjs/api/classes/animationInterface.js +1 -1
  4. package/dist/lib-cjs/api/classes/dottedPath.d.ts +79 -79
  5. package/dist/lib-cjs/api/classes/dottedPath.js +190 -190
  6. package/dist/lib-cjs/api/classes/element.d.ts +130 -125
  7. package/dist/lib-cjs/api/classes/element.js +743 -682
  8. package/dist/lib-cjs/api/classes/element.js.map +1 -1
  9. package/dist/lib-cjs/api/classes/elementParameterizable.d.ts +14 -14
  10. package/dist/lib-cjs/api/classes/elementParameterizable.js +134 -134
  11. package/dist/lib-cjs/api/classes/event.d.ts +326 -312
  12. package/dist/lib-cjs/api/classes/event.js +371 -357
  13. package/dist/lib-cjs/api/classes/event.js.map +1 -1
  14. package/dist/lib-cjs/api/classes/eventBroadcaster.d.ts +26 -26
  15. package/dist/lib-cjs/api/classes/eventBroadcaster.js +53 -53
  16. package/dist/lib-cjs/api/classes/parameter.d.ts +259 -165
  17. package/dist/lib-cjs/api/classes/parameter.js +387 -267
  18. package/dist/lib-cjs/api/classes/parameter.js.map +1 -1
  19. package/dist/lib-cjs/api/classes/parameterObservable.d.ts +36 -36
  20. package/dist/lib-cjs/api/classes/parameterObservable.js +101 -126
  21. package/dist/lib-cjs/api/classes/parameterObservable.js.map +1 -1
  22. package/dist/lib-cjs/api/classes/parameterizable.d.ts +15 -15
  23. package/dist/lib-cjs/api/classes/parameterizable.js +149 -149
  24. package/dist/lib-cjs/api/classes/placementAnimation.d.ts +43 -38
  25. package/dist/lib-cjs/api/classes/placementAnimation.js +160 -138
  26. package/dist/lib-cjs/api/classes/placementAnimation.js.map +1 -1
  27. package/dist/lib-cjs/api/classes/variant.d.ts +234 -190
  28. package/dist/lib-cjs/api/classes/variant.js +1147 -873
  29. package/dist/lib-cjs/api/classes/variant.js.map +1 -1
  30. package/dist/lib-cjs/api/classes/variantInstance.d.ts +45 -41
  31. package/dist/lib-cjs/api/classes/variantInstance.js +108 -98
  32. package/dist/lib-cjs/api/classes/variantInstance.js.map +1 -1
  33. package/dist/lib-cjs/api/classes/variantParameterizable.d.ts +17 -17
  34. package/dist/lib-cjs/api/classes/variantParameterizable.js +92 -92
  35. package/dist/lib-cjs/api/classes/viewer.d.ts +131 -128
  36. package/dist/lib-cjs/api/classes/viewer.js +567 -486
  37. package/dist/lib-cjs/api/classes/viewer.js.map +1 -1
  38. package/dist/lib-cjs/api/classes/viewerLight.d.ts +66 -65
  39. package/dist/lib-cjs/api/classes/viewerLight.js +389 -322
  40. package/dist/lib-cjs/api/classes/viewerLight.js.map +1 -1
  41. package/dist/lib-cjs/api/internal/debugViewer.d.ts +13 -13
  42. package/dist/lib-cjs/api/internal/debugViewer.js +87 -87
  43. package/dist/lib-cjs/api/internal/lensRendering.d.ts +8 -8
  44. package/dist/lib-cjs/api/internal/lensRendering.js +11 -11
  45. package/dist/lib-cjs/api/internal/sceneSetup.d.ts +10 -6
  46. package/dist/lib-cjs/api/internal/sceneSetup.js +231 -227
  47. package/dist/lib-cjs/api/internal/sceneSetup.js.map +1 -1
  48. package/dist/lib-cjs/api/manager/animationManager.d.ts +29 -29
  49. package/dist/lib-cjs/api/manager/animationManager.js +121 -121
  50. package/dist/lib-cjs/api/manager/sceneManager.d.ts +32 -32
  51. package/dist/lib-cjs/api/manager/sceneManager.js +132 -132
  52. package/dist/lib-cjs/api/manager/variantInstanceManager.d.ts +90 -90
  53. package/dist/lib-cjs/api/manager/variantInstanceManager.js +321 -321
  54. package/dist/lib-cjs/api/store/specStorage.d.ts +24 -24
  55. package/dist/lib-cjs/api/store/specStorage.js +51 -51
  56. package/dist/lib-cjs/api/util/babylonHelper.d.ts +175 -125
  57. package/dist/lib-cjs/api/util/babylonHelper.js +520 -368
  58. package/dist/lib-cjs/api/util/babylonHelper.js.map +1 -1
  59. package/dist/lib-cjs/api/util/globalTypes.d.ts +333 -279
  60. package/dist/lib-cjs/api/util/globalTypes.js +1 -1
  61. package/dist/lib-cjs/api/util/resourceHelper.d.ts +53 -51
  62. package/dist/lib-cjs/api/util/resourceHelper.js +240 -243
  63. package/dist/lib-cjs/api/util/resourceHelper.js.map +1 -1
  64. package/dist/lib-cjs/api/util/stringHelper.d.ts +9 -9
  65. package/dist/lib-cjs/api/util/stringHelper.js +25 -25
  66. package/dist/lib-cjs/buildinfo.json +3 -3
  67. package/dist/lib-cjs/index.d.ts +48 -46
  68. package/dist/lib-cjs/index.js +86 -82
  69. package/dist/lib-cjs/index.js.map +1 -1
  70. package/package.json +83 -83
  71. package/src/api/classes/animationInterface.ts +11 -11
  72. package/src/api/classes/dottedPath.ts +189 -189
  73. package/src/api/classes/element.ts +644 -617
  74. package/src/api/classes/event.ts +370 -355
  75. package/src/api/classes/eventBroadcaster.ts +54 -54
  76. package/src/api/classes/parameter.ts +408 -277
  77. package/src/api/classes/parameterObservable.ts +99 -121
  78. package/src/api/classes/{elementParameterizable.ts → parameterizable.ts} +89 -78
  79. package/src/api/classes/placementAnimation.ts +157 -133
  80. package/src/api/classes/variant.ts +817 -670
  81. package/src/api/classes/variantInstance.ts +88 -81
  82. package/src/api/classes/variantParameterizable.ts +73 -0
  83. package/src/api/classes/viewer.ts +487 -421
  84. package/src/api/classes/viewerLight.ts +330 -0
  85. package/src/api/internal/debugViewer.ts +81 -81
  86. package/src/api/internal/lensRendering.ts +10 -10
  87. package/src/api/internal/sceneSetup.ts +194 -204
  88. package/src/api/manager/animationManager.ts +116 -116
  89. package/src/api/manager/sceneManager.ts +105 -105
  90. package/src/api/manager/variantInstanceManager.ts +236 -236
  91. package/src/api/store/specStorage.ts +53 -53
  92. package/src/api/util/babylonHelper.ts +555 -392
  93. package/src/api/util/globalTypes.ts +380 -319
  94. package/src/api/util/resourceHelper.ts +184 -187
  95. package/src/api/util/stringHelper.ts +26 -0
  96. package/src/buildinfo.json +2 -2
  97. package/src/commonjs.tsconfig.json +13 -13
  98. package/src/declaration.tsconfig.json +10 -10
  99. package/src/dev.ts +42 -46
  100. package/src/es6.tsconfig.json +13 -13
  101. package/src/index.ts +91 -87
  102. package/src/pagesconfig.json +61 -57
  103. package/src/tsconfig.json +43 -43
  104. package/src/tsconfig.types.json +9 -9
  105. package/src/types.d.ts +4 -4
  106. package/dist/webpack-stats.json +0 -0
@@ -1,237 +1,237 @@
1
- import { merge } from 'lodash-es';
2
- import { DottedPath } from '../classes/dottedPath';
3
- import { Event } from '../classes/event';
4
- import { EventBroadcaster } from '../classes/eventBroadcaster';
5
- import { Parameter } from '../classes/parameter';
6
- import { Variant } from '../classes/variant';
7
- import { VariantInstance } from '../classes/variantInstance';
8
-
9
- /**
10
- * There's usually no need to instantiate a {@link VariantInstanceManager} yourself as it is provided by each
11
- * {@link Viewer} instance via its {@link Viewer.variantInstances}
12
- */
13
- export class VariantInstanceManager extends EventBroadcaster {
14
-
15
- protected variantInstances: Map<string, VariantInstance> = new Map();
16
-
17
- protected variantInstanceDefinitions: Map<string, VariantInstanceDefinition> = new Map();
18
-
19
- protected parameters: ParameterBag = {};
20
-
21
- protected queues: { [name: string]: Promise<VariantInstance>[] } = {};
22
-
23
- /**
24
- * Constructor.
25
- */
26
- protected constructor( protected rootVariant: Variant ) {
27
- super();
28
- }
29
-
30
- /**
31
- * Creates a {@link VariantInstanceManager} based on given root Variant.
32
- */
33
- public static async create( rootVariant: Variant ): Promise<VariantInstanceManager> {
34
- return new VariantInstanceManager( rootVariant );
35
- }
36
-
37
- /**
38
- * Gets all instances.
39
- */
40
- get all() {
41
- const all: VariantInstance[] = [];
42
- this.variantInstances.forEach( variantInstance => all.push( variantInstance ) );
43
- return all;
44
- }
45
-
46
- /**
47
- * Gets all instance definitions.
48
- */
49
- get allDefinitions() {
50
- const all: VariantInstanceDefinition[] = [];
51
- this.variantInstanceDefinitions.forEach( definition => all.push( definition ) );
52
- return all;
53
- }
54
-
55
- /**
56
- * Checks whether a {@link VariantInstance} is either registered or created.
57
- */
58
- public has( name: string ) {
59
- return this.variantInstances.has( name ) || this.variantInstanceDefinitions.has( name );
60
- }
61
-
62
- /**
63
- * Retrieves a {@link VariantInstance}.
64
- * If the instance has been created, the created instance will be returned.
65
- * If the instance is registered, the instance will be created.
66
- * If none of these apply, an Error is thrown.
67
- *
68
- * @emits {@link Event.VARIANT_INSTANCE_CREATED}
69
- */
70
- public async get( name: string ): Promise<VariantInstance> {
71
- if( this.variantInstances.has( name ) ) {
72
- return this.variantInstances.get( name )!;
73
- }
74
- if( this.variantInstanceDefinitions.has( name ) ) {
75
- const definition = this.variantInstanceDefinitions.get( name )!;
76
- return await this.createFromDefinition( definition );
77
- }
78
- throw Error( `VariantInstance with name "${name}" neither created nor configured.` );
79
- }
80
-
81
- /**
82
- * Creates and registers a unique {@link VariantInstance}.
83
- *
84
- * @emits {@link Event.VARIANT_INSTANCE_CREATED}
85
- */
86
- public async create( dottedPath: DottedPathArgument, name?: string, parameters?: ParameterBag ): Promise<VariantInstance> {
87
- const variant = DottedPath.create( dottedPath ).path;
88
- const definition = {
89
- name: this.ensureUniqueName( name ? name : variant ),
90
- variant: variant,
91
- parameters: parameters
92
- };
93
- return await this.createFromDefinition( definition );
94
- }
95
-
96
- /**
97
- * Registers a unique {@link VariantInstanceDefinition}.
98
- *
99
- * @emits {@link Event.VARIANT_INSTANCE_REGISTERED}
100
- */
101
- public register( instanceDefinition: VariantInstanceDefinition ): VariantInstanceDefinition {
102
- const name = instanceDefinition.name ? instanceDefinition.name
103
- : DottedPath.create( instanceDefinition.variant ).path;
104
- instanceDefinition.name = this.ensureUniqueName( name );
105
- this.variantInstanceDefinitions.set( instanceDefinition.name, instanceDefinition );
106
- this.broadcastEvent( Event.VARIANT_INSTANCE_REGISTERED, instanceDefinition );
107
- return instanceDefinition;
108
- }
109
-
110
- /**
111
- * Commits the {@link ParameterBag} to all registered and created {@link VariantInstance}s.
112
- */
113
- public async commitParameters( parameters: ParameterBag = {}, onlyVisible: boolean = true ): Promise<VariantInstanceManager> {
114
- this.parameters = merge(this.parameters, parameters);
115
- this.allDefinitions.forEach( definition => definition.parameters = merge( definition.parameters, this.parameters ) );
116
- let instances = this.all;
117
- if( onlyVisible ) {
118
- instances = this.all.filter( instance => instance.variant.visible );
119
- }
120
- const promises = instances.map( instance => instance.variant.commitParameters( this.parameters ) );
121
- await Promise.all( promises );
122
- return this;
123
- }
124
-
125
- /**
126
- * Show one {@link VariantInstance}. If `exclusive` is set to `true` it will hide all other
127
- * instances handled by this {@link VariantInstanceManager}. If the function is called synchronously,
128
- * the calls end up in a queue according to the FIFO principle (the last one wins).
129
- */
130
- public async show( name: string, exclusive: boolean = false ): Promise<VariantInstance> {
131
- const promise: Promise<VariantInstance> = new Promise(async resolve => {
132
- if( exclusive ) {
133
- this.allDefinitions.filter( definition => definition.name !== name ).forEach( definition => {
134
- const parameters: ParameterBag = {};
135
- parameters[Parameter.VISIBLE] = false;
136
- definition.parameters = merge( definition.parameters, parameters);
137
- });
138
- await Promise.all( this.all.filter( _instance => _instance.name !== name ).map( _instance => {
139
- return _instance.variant.hide();
140
- } ) );
141
- }
142
- const instance = await this.get( name );
143
- await instance.variant.show();
144
- resolve(instance);
145
- });
146
- if( ! this.queues['show'] ) {
147
- this.queues['show'] = [];
148
- }
149
- this.queues['show'].push(promise);
150
- for( let i = 0; i < this.queues['show'].length; i++) {
151
- const instance = await this.queues['show'][i];
152
- if( exclusive ) {
153
- if( i + 1 < this.queues['show'].length ) {
154
- await instance.variant.hide();
155
- }
156
- }
157
- }
158
- this.queues['show'] = [];
159
- return promise;
160
- }
161
-
162
- /**
163
- * Clones and registers a new {@link VariantInstance} with a new, unique name.
164
- *
165
- * @emits {@link Event.VARIANT_INSTANCE_CLONED}
166
- */
167
- public async clone( name: string, newName: string, parameters?: ParameterBag ): Promise<VariantInstance> {
168
- if( !this.variantInstances.has( name ) ) {
169
- throw Error( `VariantInstance with name "${name}" does not exist.` );
170
- }
171
- const variantInstance = this.variantInstances.get( name )!;
172
- const definition = {
173
- name: this.ensureUniqueName( newName ? newName : name ),
174
- variant: variantInstance.variant.id,
175
- parameters: parameters
176
- };
177
- const living = await variantInstance.variant.createLiving( definition.parameters );
178
- const variantInstanceClone = new VariantInstance( living, definition.name );
179
- this.variantInstances.set( definition.name, variantInstanceClone );
180
- variantInstance.broadcastEvent( Event.VARIANT_INSTANCE_CLONED, variantInstanceClone );
181
- return variantInstanceClone;
182
- }
183
-
184
- /**
185
- * Destroys the instance by removing the {@link Variant} and de-registering the instance.
186
- *
187
- * @emits {@link Event.VARIANT_INSTANCE_DESTROYED}
188
- */
189
- public destroy( name: string ): VariantInstanceManager {
190
- if( !this.variantInstances.has( name ) ) {
191
- throw Error( `VariantInstance with name "${name}" does not exist.` );
192
- }
193
- const variantInstance = this.variantInstances.get( name )!;
194
- variantInstance.variant.destroy();
195
- this.variantInstances.delete( name );
196
- variantInstance.broadcastEvent( Event.VARIANT_INSTANCE_DESTROYED, variantInstance );
197
- return this;
198
- }
199
-
200
- /**
201
- * Creates and registers a unique {@link VariantInstance} from `definition`.
202
- *
203
- * @emits {@link Event.VARIANT_INSTANCE_CREATED}
204
- */
205
- protected async createFromDefinition( definition: VariantInstanceDefinition ): Promise<VariantInstance> {
206
- if( !definition.name ) {
207
- throw new Error( `Given VariantInstanceDefinition ${definition.variant} is missing a name property.` );
208
- }
209
- if( this.variantInstances.has( definition.name ) ) {
210
- throw new Error( `VariantInstance with name "${definition.name}" is already created.` );
211
- }
212
- if( !this.rootVariant ) {
213
- throw new Error( `Viewer has not been properly bootstrapped (rootVariant missing).` );
214
- }
215
- const variant = await this.rootVariant.getDescendant( definition.variant );
216
- const living = await variant.createLiving( definition.parameters );
217
- const variantInstance = new VariantInstance( living, definition.name );
218
- this.variantInstances.set( definition.name, variantInstance );
219
- this.broadcastEvent( Event.VARIANT_INSTANCE_CREATED, variantInstance );
220
- return variantInstance;
221
- }
222
-
223
- /**
224
- * Creates a unique instance name based on already registered instance names on the {@link VariantInstanceManager}.
225
- * Appends an `_i` to the name where `i = 0` and increments as long as the name is not registered.
226
- */
227
- protected ensureUniqueName( name: string ): string {
228
- let instanceName = name;
229
- let i = 1;
230
- while( this.has( instanceName ) ) {
231
- instanceName = `${name}_${i}`;
232
- i++;
233
- }
234
- return instanceName;
235
- }
236
-
1
+ import { merge } from 'lodash-es';
2
+ import { DottedPath } from '../classes/dottedPath';
3
+ import { Event } from '../classes/event';
4
+ import { EventBroadcaster } from '../classes/eventBroadcaster';
5
+ import { Parameter } from '../classes/parameter';
6
+ import { Variant } from '../classes/variant';
7
+ import { VariantInstance } from '../classes/variantInstance';
8
+
9
+ /**
10
+ * There's usually no need to instantiate a {@link VariantInstanceManager} yourself as it is provided by each
11
+ * {@link Viewer} instance via its {@link Viewer.variantInstances}
12
+ */
13
+ export class VariantInstanceManager extends EventBroadcaster {
14
+
15
+ protected variantInstances: Map<string, VariantInstance> = new Map();
16
+
17
+ protected variantInstanceDefinitions: Map<string, VariantInstanceDefinition> = new Map();
18
+
19
+ protected parameters: ParameterBag = {};
20
+
21
+ protected queues: { [name: string]: Promise<VariantInstance>[] } = {};
22
+
23
+ /**
24
+ * Constructor.
25
+ */
26
+ protected constructor( protected rootVariant: Variant ) {
27
+ super();
28
+ }
29
+
30
+ /**
31
+ * Creates a {@link VariantInstanceManager} based on given root Variant.
32
+ */
33
+ public static async create( rootVariant: Variant ): Promise<VariantInstanceManager> {
34
+ return new VariantInstanceManager( rootVariant );
35
+ }
36
+
37
+ /**
38
+ * Gets all instances.
39
+ */
40
+ get all() {
41
+ const all: VariantInstance[] = [];
42
+ this.variantInstances.forEach( variantInstance => all.push( variantInstance ) );
43
+ return all;
44
+ }
45
+
46
+ /**
47
+ * Gets all instance definitions.
48
+ */
49
+ get allDefinitions() {
50
+ const all: VariantInstanceDefinition[] = [];
51
+ this.variantInstanceDefinitions.forEach( definition => all.push( definition ) );
52
+ return all;
53
+ }
54
+
55
+ /**
56
+ * Checks whether a {@link VariantInstance} is either registered or created.
57
+ */
58
+ public has( name: string ) {
59
+ return this.variantInstances.has( name ) || this.variantInstanceDefinitions.has( name );
60
+ }
61
+
62
+ /**
63
+ * Retrieves a {@link VariantInstance}.
64
+ * If the instance has been created, the created instance will be returned.
65
+ * If the instance is registered, the instance will be created.
66
+ * If none of these apply, an Error is thrown.
67
+ *
68
+ * @emits {@link Event.VARIANT_INSTANCE_CREATED}
69
+ */
70
+ public async get( name: string ): Promise<VariantInstance> {
71
+ if( this.variantInstances.has( name ) ) {
72
+ return this.variantInstances.get( name )!;
73
+ }
74
+ if( this.variantInstanceDefinitions.has( name ) ) {
75
+ const definition = this.variantInstanceDefinitions.get( name )!;
76
+ return await this.createFromDefinition( definition );
77
+ }
78
+ throw Error( `VariantInstance with name "${name}" neither created nor configured.` );
79
+ }
80
+
81
+ /**
82
+ * Creates and registers a unique {@link VariantInstance}.
83
+ *
84
+ * @emits {@link Event.VARIANT_INSTANCE_CREATED}
85
+ */
86
+ public async create( dottedPath: DottedPathArgument, name?: string, parameters?: ParameterBag ): Promise<VariantInstance> {
87
+ const variant = DottedPath.create( dottedPath ).path;
88
+ const definition = {
89
+ name: this.ensureUniqueName( name ? name : variant ),
90
+ variant: variant,
91
+ parameters: parameters
92
+ };
93
+ return await this.createFromDefinition( definition );
94
+ }
95
+
96
+ /**
97
+ * Registers a unique {@link VariantInstanceDefinition}.
98
+ *
99
+ * @emits {@link Event.VARIANT_INSTANCE_REGISTERED}
100
+ */
101
+ public register( instanceDefinition: VariantInstanceDefinition ): VariantInstanceDefinition {
102
+ const name = instanceDefinition.name ? instanceDefinition.name
103
+ : DottedPath.create( instanceDefinition.variant ).path;
104
+ instanceDefinition.name = this.ensureUniqueName( name );
105
+ this.variantInstanceDefinitions.set( instanceDefinition.name, instanceDefinition );
106
+ this.broadcastEvent( Event.VARIANT_INSTANCE_REGISTERED, instanceDefinition );
107
+ return instanceDefinition;
108
+ }
109
+
110
+ /**
111
+ * Commits the {@link ParameterBag} to all registered and created {@link VariantInstance}s.
112
+ */
113
+ public async commitParameters( parameters: ParameterBag = {}, onlyVisible: boolean = true ): Promise<VariantInstanceManager> {
114
+ this.parameters = merge(this.parameters, parameters);
115
+ this.allDefinitions.forEach( definition => definition.parameters = merge( definition.parameters, this.parameters ) );
116
+ let instances = this.all;
117
+ if( onlyVisible ) {
118
+ instances = this.all.filter( instance => instance.variant.visible );
119
+ }
120
+ const promises = instances.map( instance => instance.variant.commitParameters( this.parameters ) );
121
+ await Promise.all( promises );
122
+ return this;
123
+ }
124
+
125
+ /**
126
+ * Show one {@link VariantInstance}. If `exclusive` is set to `true` it will hide all other
127
+ * instances handled by this {@link VariantInstanceManager}. If the function is called synchronously,
128
+ * the calls end up in a queue according to the FIFO principle (the last one wins).
129
+ */
130
+ public async show( name: string, exclusive: boolean = false ): Promise<VariantInstance> {
131
+ const promise: Promise<VariantInstance> = new Promise(async resolve => {
132
+ if( exclusive ) {
133
+ this.allDefinitions.filter( definition => definition.name !== name ).forEach( definition => {
134
+ const parameters: ParameterBag = {};
135
+ parameters[Parameter.VISIBLE] = false;
136
+ definition.parameters = merge( definition.parameters, parameters);
137
+ });
138
+ await Promise.all( this.all.filter( _instance => _instance.name !== name ).map( _instance => {
139
+ return _instance.variant.hide();
140
+ } ) );
141
+ }
142
+ const instance = await this.get( name );
143
+ await instance.variant.show();
144
+ resolve(instance);
145
+ });
146
+ if( ! this.queues['show'] ) {
147
+ this.queues['show'] = [];
148
+ }
149
+ this.queues['show'].push(promise);
150
+ for( let i = 0; i < this.queues['show'].length; i++) {
151
+ const instance = await this.queues['show'][i];
152
+ if( exclusive ) {
153
+ if( i + 1 < this.queues['show'].length ) {
154
+ await instance.variant.hide();
155
+ }
156
+ }
157
+ }
158
+ this.queues['show'] = [];
159
+ return promise;
160
+ }
161
+
162
+ /**
163
+ * Clones and registers a new {@link VariantInstance} with a new, unique name.
164
+ *
165
+ * @emits {@link Event.VARIANT_INSTANCE_CLONED}
166
+ */
167
+ public async clone( name: string, newName: string, parameters?: ParameterBag ): Promise<VariantInstance> {
168
+ if( !this.variantInstances.has( name ) ) {
169
+ throw Error( `VariantInstance with name "${name}" does not exist.` );
170
+ }
171
+ const variantInstance = this.variantInstances.get( name )!;
172
+ const definition = {
173
+ name: this.ensureUniqueName( newName ? newName : name ),
174
+ variant: variantInstance.variant.id,
175
+ parameters: parameters
176
+ };
177
+ const living = await variantInstance.variant.createLiving( definition.parameters );
178
+ const variantInstanceClone = new VariantInstance( living, definition.name );
179
+ this.variantInstances.set( definition.name, variantInstanceClone );
180
+ variantInstance.broadcastEvent( Event.VARIANT_INSTANCE_CLONED, variantInstanceClone );
181
+ return variantInstanceClone;
182
+ }
183
+
184
+ /**
185
+ * Destroys the instance by removing the {@link Variant} and de-registering the instance.
186
+ *
187
+ * @emits {@link Event.VARIANT_INSTANCE_DESTROYED}
188
+ */
189
+ public destroy( name: string ): VariantInstanceManager {
190
+ if( !this.variantInstances.has( name ) ) {
191
+ throw Error( `VariantInstance with name "${name}" does not exist.` );
192
+ }
193
+ const variantInstance = this.variantInstances.get( name )!;
194
+ variantInstance.variant.destroy();
195
+ this.variantInstances.delete( name );
196
+ variantInstance.broadcastEvent( Event.VARIANT_INSTANCE_DESTROYED, variantInstance );
197
+ return this;
198
+ }
199
+
200
+ /**
201
+ * Creates and registers a unique {@link VariantInstance} from `definition`.
202
+ *
203
+ * @emits {@link Event.VARIANT_INSTANCE_CREATED}
204
+ */
205
+ protected async createFromDefinition( definition: VariantInstanceDefinition ): Promise<VariantInstance> {
206
+ if( !definition.name ) {
207
+ throw new Error( `Given VariantInstanceDefinition ${definition.variant} is missing a name property.` );
208
+ }
209
+ if( this.variantInstances.has( definition.name ) ) {
210
+ throw new Error( `VariantInstance with name "${definition.name}" is already created.` );
211
+ }
212
+ if( !this.rootVariant ) {
213
+ throw new Error( `Viewer has not been properly bootstrapped (rootVariant missing).` );
214
+ }
215
+ const variant = await this.rootVariant.getDescendant( definition.variant );
216
+ const living = await variant.createLiving( definition.parameters );
217
+ const variantInstance = new VariantInstance( living, definition.name );
218
+ this.variantInstances.set( definition.name, variantInstance );
219
+ this.broadcastEvent( Event.VARIANT_INSTANCE_CREATED, variantInstance );
220
+ return variantInstance;
221
+ }
222
+
223
+ /**
224
+ * Creates a unique instance name based on already registered instance names on the {@link VariantInstanceManager}.
225
+ * Appends an `_i` to the name where `i = 0` and increments as long as the name is not registered.
226
+ */
227
+ protected ensureUniqueName( name: string ): string {
228
+ let instanceName = name;
229
+ let i = 1;
230
+ while( this.has( instanceName ) ) {
231
+ instanceName = `${name}_${i}`;
232
+ i++;
233
+ }
234
+ return instanceName;
235
+ }
236
+
237
237
  }
@@ -1,54 +1,54 @@
1
- import { get, set } from 'lodash-es';
2
-
3
- export class SpecStorage {
4
-
5
- public static instance: SpecStorage|undefined;
6
-
7
- /**
8
- * Constructor.
9
- */
10
- protected constructor( protected spec: StructureJson ) {
11
- }
12
-
13
- /**
14
- * Creates {@link SpecStorage} from spec.
15
- */
16
- public static createFromSpec( structureJson: StructureJson ): SpecStorage {
17
- if( this.instance ) {
18
- throw new Error( `SpecStorage has already been created.` );
19
- }
20
- return (this.instance = new this( structureJson ));
21
- }
22
-
23
- /**
24
- * Destroys the current instance to allow reinstantiating the SpecStorage.
25
- */
26
- public static destroy() {
27
- this.instance = undefined
28
- }
29
-
30
- /**
31
- * Gets spec from key.
32
- */
33
- public static get<T>( key?: string ): T {
34
- if( !this.instance || !this.instance.spec ) {
35
- throw new Error( `SpecStore has not been properly created.` );
36
- }
37
- if( key ) {
38
- return get( this.instance.spec, key ) as T;
39
- }
40
- return this.instance.spec as T;
41
- }
42
-
43
- /**
44
- * Sets value for spec for given key.
45
- */
46
- public static set<T>( key: string, value: T ) {
47
- if( !this.instance || !this.instance.spec ) {
48
- throw new Error( `SpecStore has not been properly created.` );
49
- }
50
- set( this.instance.spec, key, value );
51
- return SpecStorage;
52
- }
53
-
1
+ import { get, set } from 'lodash-es';
2
+
3
+ export class SpecStorage {
4
+
5
+ public static instance: SpecStorage|undefined;
6
+
7
+ /**
8
+ * Constructor.
9
+ */
10
+ protected constructor( protected spec: StructureJson ) {
11
+ }
12
+
13
+ /**
14
+ * Creates {@link SpecStorage} from spec.
15
+ */
16
+ public static createFromSpec( structureJson: StructureJson ): SpecStorage {
17
+ if( this.instance ) {
18
+ throw new Error( `SpecStorage has already been created.` );
19
+ }
20
+ return (this.instance = new this( structureJson ));
21
+ }
22
+
23
+ /**
24
+ * Destroys the current instance to allow reinstantiating the SpecStorage.
25
+ */
26
+ public static destroy() {
27
+ this.instance = undefined
28
+ }
29
+
30
+ /**
31
+ * Gets spec from key.
32
+ */
33
+ public static get<T>( key?: string ): T {
34
+ if( !this.instance || !this.instance.spec ) {
35
+ throw new Error( `SpecStore has not been properly created.` );
36
+ }
37
+ if( key ) {
38
+ return get( this.instance.spec, key ) as T;
39
+ }
40
+ return this.instance.spec as T;
41
+ }
42
+
43
+ /**
44
+ * Sets value for spec for given key.
45
+ */
46
+ public static set<T>( key: string, value: T ) {
47
+ if( !this.instance || !this.instance.spec ) {
48
+ throw new Error( `SpecStore has not been properly created.` );
49
+ }
50
+ set( this.instance.spec, key, value );
51
+ return SpecStorage;
52
+ }
53
+
54
54
  }