src/loaders/phoenix-loader.ts
Loader for processing and loading an event.
Properties |
|
Methods |
|
constructor()
|
Defined in src/loaders/phoenix-loader.ts:33
|
Create the Phoenix loader. |
Protected eventData |
Type : any
|
Defined in src/loaders/phoenix-loader.ts:27
|
Event data processed by the loader. |
Private graphicsLibrary |
Type : ThreeManager
|
Defined in src/loaders/phoenix-loader.ts:23
|
ThreeService to perform three.js related functions. |
Protected labelsObject |
Type : literal type
|
Default value : {}
|
Defined in src/loaders/phoenix-loader.ts:33
|
Object containing event object labels. |
Protected loadingManager |
Type : LoadingManager
|
Defined in src/loaders/phoenix-loader.ts:29
|
Loading manager for loadable resources |
Protected stateManager |
Type : StateManager
|
Defined in src/loaders/phoenix-loader.ts:31
|
Loading manager for loadable resources |
Private ui |
Type : UIManager
|
Defined in src/loaders/phoenix-loader.ts:25
|
UIService to perform UI related functions. |
Private addCollection | ||||||||||||||||||||||||||||
addCollection(objectCollection: any, collectionName: string, getObject: (object: any,typeName: string) => void, typeName: string, objectGroup: Group, concatonateObjs: boolean)
|
||||||||||||||||||||||||||||
Defined in src/loaders/phoenix-loader.ts:516
|
||||||||||||||||||||||||||||
Adds to the event display all the objects inside a collection.
Parameters :
Returns :
void
|
Public addLabelToEventObject | ||||||||||||||||
addLabelToEventObject(label: string, collection: string, indexInCollection: number)
|
||||||||||||||||
Defined in src/loaders/phoenix-loader.ts:733
|
||||||||||||||||
Add label of event object to the labels object.
Parameters :
Returns :
string
A unique label ID string. |
Protected addObjectType | |||||||||||||||||||||||||||||||||||
addObjectType(object: any, getObject: any, typeName: string, concatonateObjs: boolean, cuts?: Cut[], extendEventDataTypeUI?: (typeFolder?: GUI,typeFolderPM?: PhoenixMenuNode) => void)
|
|||||||||||||||||||||||||||||||||||
Defined in src/loaders/phoenix-loader.ts:448
|
|||||||||||||||||||||||||||||||||||
Adds to the event display all collections of a given object type.
Parameters :
Returns :
void
|
Public addScaleOptions | ||||||||||||||||
addScaleOptions(configKey: string, configLabel: string, scaleFunction: (value: number) => void)
|
||||||||||||||||
Defined in src/loaders/phoenix-loader.ts:770
|
||||||||||||||||
Get function to add options to scale event object type by.
Parameters :
Function which adds scale options to Phoenix menu and dat.GUI menu. |
Public buildEventData | ||||||||||||||||||||
buildEventData(eventData: any, graphicsLibrary: ThreeManager, ui: UIManager, infoLogger: InfoLogger)
|
||||||||||||||||||||
Defined in src/loaders/phoenix-loader.ts:51
|
||||||||||||||||||||
Takes an object that represents ONE event and takes care of adding the different objects to the graphics library and the UI controls.
Parameters :
Returns :
void
|
Public getCollection | ||||||||
getCollection(collectionName: string)
|
||||||||
Defined in src/loaders/phoenix-loader.ts:127
|
||||||||
Get the collection with the given collection name from the event data.
Parameters :
Returns :
any
An object containing the collection. |
Public getCollections |
getCollections()
|
Defined in src/loaders/phoenix-loader.ts:103
|
Get list of collections in the event data.
Returns :
string[]
List of all collection names. |
Protected getCompound | |||||||||||||||||||||||||
getCompound(params: any, name: string, objectIsTrack: boolean, objectIsCluster: boolean)
|
|||||||||||||||||||||||||
Defined in src/loaders/phoenix-loader.ts:581
|
|||||||||||||||||||||||||
Process the compound object (e.g. Muon, Electron, Photon) from the given parameters and get it as a group. FIXME. This is currently here and not in PhoenixObjects because we need to handle linked objects.
Parameters :
Returns :
Object3D
Muon group containing Clusters and Tracks. |
Protected getCompoundCluster |
getCompoundCluster(params: any, name: string)
|
Defined in src/loaders/phoenix-loader.ts:571
|
Process the compound object of cluster type (e.g. Photon, ..) from the given parameters and get it as a group.
Returns :
Object3D
|
Protected getCompoundTrack |
getCompoundTrack(params: any, name: string)
|
Defined in src/loaders/phoenix-loader.ts:565
|
Process the compound object of track type (e.g. Muon, Electron, ..) from the given parameters and get it as a group.
Returns :
Object3D
|
getEventMetadata |
getEventMetadata()
|
Defined in src/loaders/phoenix-loader.ts:679
|
Get metadata associated to the event (experiment info, time, run, event...).
Returns :
any[]
Metadata of the event. |
Public getEventsList | ||||||||
getEventsList(eventsData: any)
|
||||||||
Defined in src/loaders/phoenix-loader.ts:87
|
||||||||
Get the list of event names from the event data.
Parameters :
Returns :
string[]
List of event names. |
Public getLabelsObject |
getLabelsObject()
|
Defined in src/loaders/phoenix-loader.ts:759
|
Get the object containing labels.
Returns :
literal type
The labels object. |
Private getObjectTypeCollections | ||||||||
getObjectTypeCollections(object: any)
|
||||||||
Defined in src/loaders/phoenix-loader.ts:551
|
||||||||
Get collection names of a given object type.
Parameters :
Returns :
string[]
List of collection names of an object type (Tracks, Jets, CaloClusters etc.). |
Protected loadObjectTypes | ||||||||
loadObjectTypes(eventData: any)
|
||||||||
Defined in src/loaders/phoenix-loader.ts:148
|
||||||||
Receives an object containing the data from an event and parses it to reconstruct the different collections of physics objects.
Parameters :
Returns :
void
|
import { Group, Object3D, Vector3 } from 'three';
import { GUI } from 'dat.gui';
import type { EventDataLoader } from './event-data-loader';
import { UIManager } from '../managers/ui-manager/index';
import { ThreeManager } from '../managers/three-manager/index';
import { Cut } from '../lib/models/cut.model';
import { PhoenixObjects } from './objects/phoenix-objects';
import { InfoLogger } from '../helpers/info-logger';
import { PhoenixMenuNode } from '../managers/ui-manager/phoenix-menu/phoenix-menu-node';
import { LoadingManager } from '../managers/loading-manager';
import { StateManager } from '../managers/state-manager';
import { CoordinateHelper } from '../helpers/coordinate-helper';
import { getLabelTitle } from '../helpers/labels';
import { DatGUIMenuUI } from '../managers/ui-manager/dat-gui-ui';
import { PhoenixMenuUI } from '../managers/ui-manager/phoenix-menu/phoenix-menu-ui';
import * as _ from 'lodash';
/**
* Loader for processing and loading an event.
*/
export class PhoenixLoader implements EventDataLoader {
/** ThreeService to perform three.js related functions. */
private graphicsLibrary: ThreeManager;
/** UIService to perform UI related functions. */
private ui: UIManager;
/** Event data processed by the loader. */
protected eventData: any;
/** Loading manager for loadable resources */
protected loadingManager: LoadingManager;
/** Loading manager for loadable resources */
protected stateManager: StateManager;
/** Object containing event object labels. */
protected labelsObject: { [key: string]: any } = {};
/**
* Create the Phoenix loader.
*/
constructor() {
this.loadingManager = new LoadingManager();
this.stateManager = new StateManager();
}
/**
* Takes an object that represents ONE event and takes care of adding
* the different objects to the graphics library and the UI controls.
* @param eventData Object representing the event.
* @param graphicsLibrary Service containing functionality to draw the 3D objects.
* @param ui Service for showing menus and controls to manipulate the geometries.
* @param infoLogger Service for logging data to the information panel.
*/
public buildEventData(
eventData: any,
graphicsLibrary: ThreeManager,
ui: UIManager,
infoLogger: InfoLogger,
): void {
this.graphicsLibrary = graphicsLibrary;
this.ui = ui;
this.eventData = eventData;
// Replacing tracks with tracks through Runge-Kutta
// TODO - make this configurable? Or possibly automatic if tracks have <2 positions to draw?
// Object.assign(this.eventData.Tracks, this.getTracksWithRungeKutta(this.eventData['Tracks']));
// initiate load
this.loadObjectTypes(eventData);
const eventNumber = eventData['event number']
? eventData['event number']
: eventData['eventNumber'];
const runNumber = eventData['run number']
? eventData['run number']
: eventData['runNumber'];
infoLogger.add('Event#' + eventNumber + ' from run#' + runNumber, 'Loaded');
this.stateManager.eventMetadata = {
runNumber,
eventNumber,
};
}
/**
* Get the list of event names from the event data.
* @param eventsData Object containing all event data.
* @returns List of event names.
*/
public getEventsList(eventsData: any): string[] {
const eventsList: string[] = [];
for (const eventName in eventsData) {
if (eventsData[eventName] !== null) {
eventsList.push(eventName);
}
}
return eventsList;
}
/**
* Get list of collections in the event data.
* @returns List of all collection names.
*/
public getCollections(): string[] {
if (!this.eventData) {
return [];
}
const collections = [];
for (const objectType in this.eventData) {
if (
this.eventData[objectType] &&
typeof this.eventData[objectType] === 'object'
) {
for (const collection in this.eventData[objectType]) {
collections.push(collection);
}
}
}
return collections;
}
/**
* Get the collection with the given collection name from the event data.
* @param collectionName Name of the collection to get.
* @returns An object containing the collection.
*/
public getCollection(collectionName: string): any {
if (!this.eventData) {
return null;
}
for (const objectType in this.eventData) {
if (this.eventData[objectType]) {
for (const collection in this.eventData[objectType]) {
if (collection === collectionName) {
return this.eventData[objectType][collection];
}
}
}
}
}
/**
* Receives an object containing the data from an event and parses it
* to reconstruct the different collections of physics objects.
* @param eventData Representing ONE event (expressed in the Phoenix format).
*/
protected loadObjectTypes(eventData: any) {
// PI with 2 fraction digits
const pi = parseFloat(Math.PI.toFixed(2));
if (eventData.Tracks) {
// (Optional) Cuts can be added to any physics object.
const cuts: Cut[] = [
new Cut('phi', -pi, pi, 0.01),
new Cut('eta', -4, 4, 0.1),
new Cut('chi2', 0, 100),
new Cut('dof', 0, 100),
new Cut('pT', 0, 50000, 0.1),
new Cut('z0', -30, 30, 0.1),
new Cut('d0', -30, 30, 0.1),
];
this.addObjectType(
eventData.Tracks,
PhoenixObjects.getTrack,
'Tracks',
false,
cuts,
);
}
if (eventData.Jets) {
// (Optional) Cuts can be added to any physics object.
const cuts = [
new Cut('phi', -pi, pi, 0.01),
new Cut('eta', -5.0, 5.0, 0.1),
new Cut('energy', 0, 600000, 100),
];
const scaleJets = (value: number) => {
this.graphicsLibrary.getSceneManager().scaleJets(value);
};
const addJetsSizeOption = this.addScaleOptions(
'jetsScale',
'Jets Scale',
scaleJets,
);
this.addObjectType(
eventData.Jets,
PhoenixObjects.getJet,
'Jets',
false,
cuts,
addJetsSizeOption,
);
}
if (eventData.Hits) {
// Cannot currently cut on just a position array.
this.addObjectType(eventData.Hits, PhoenixObjects.getHits, 'Hits', true);
}
if (eventData.CaloClusters) {
// (Optional) Cuts can be added to any physics object.
const cuts = [
new Cut('phi', -pi, pi, 0.01),
new Cut('eta', -5.0, 5.0, 0.1),
new Cut('energy', 0, 10000),
];
const scaleCaloClusters = (value: number) => {
this.graphicsLibrary
.getSceneManager()
.scaleChildObjects('CaloClusters', value, 'z');
};
const addCaloClusterOptions = this.addScaleOptions(
'caloClustersScale',
'CaloClusters Scale',
scaleCaloClusters,
);
this.addObjectType(
eventData.CaloClusters,
PhoenixObjects.getCluster,
'CaloClusters',
false,
cuts,
addCaloClusterOptions,
);
}
if (eventData.CaloCells) {
// (Optional) Cuts can be added to any physics object.
const cuts = [
new Cut('phi', -pi, pi, 0.01),
new Cut('eta', -5.0, 5.0, 0.1),
new Cut('energy', 0, 10000),
];
const scaleCaloCells = (value: number) => {
this.graphicsLibrary
.getSceneManager()
.scaleChildObjects('CaloCells', value, 'z');
};
const addCaloCellOptions = this.addScaleOptions(
'caloCellsScale',
'CaloCells Scale',
scaleCaloCells,
);
// FIXME! Need to pass the radius in.
this.addObjectType(
eventData.CaloCells,
PhoenixObjects.getCaloCell,
'CaloCells',
false,
cuts,
addCaloCellOptions,
);
}
if (eventData.PlanarCaloCells) {
// (Optional) Cuts can be added to any physics object.
const cuts = [new Cut('energy', 0, 10000)];
const scalePlanarCaloCells = (value: number) => {
this.graphicsLibrary
.getSceneManager()
.scaleChildObjects('PlanarCaloCells', value, 'z');
};
const addPlanarCaloCellsOptions = this.addScaleOptions(
'planarCaloCellsScale',
'PlanarCaloCells Scale',
scalePlanarCaloCells,
);
const collections: { [key: string]: any } = {};
for (const collectionName in eventData.PlanarCaloCells) {
const collection = eventData.PlanarCaloCells[collectionName];
const plane = collection['plane'];
const unitVector = new Vector3(...plane.slice(0, 3)).normalize();
collection['cells'].forEach(
(cell: any) => (cell['plane'] = [...unitVector.toArray(), plane[3]]),
);
collections[collectionName] = collection['cells'];
}
this.addObjectType(
collections,
PhoenixObjects.getPlanarCaloCell,
'PlanarCaloCells',
false,
cuts,
addPlanarCaloCellsOptions,
);
}
if (eventData.IrregularCaloCells) {
// (Optional) Cuts can be added to any physics object.
const cuts = [new Cut('layer', 0, 10), new Cut('energy', 0, 10000)];
const scaleIrregularCaloCells = (value: number) => {
this.graphicsLibrary
.getSceneManager()
.scaleChildObjects('IrregularCaloCells', value, 'z');
};
const addIrregularCaloCellOptions = this.addScaleOptions(
'IrregularCaloCellsScale',
'IrregularCaloCells Scale',
scaleIrregularCaloCells,
);
this.addObjectType(
eventData.IrregularCaloCells,
PhoenixObjects.getIrregularCaloCell,
'IrregularCaloCells',
false,
cuts,
addIrregularCaloCellOptions,
);
}
if (eventData.Muons) {
const cuts = [
new Cut('phi', -pi, pi, 0.01),
new Cut('eta', -4, 4, 0.1),
new Cut('energy', 0, 10000),
new Cut('pT', 0, 50000),
];
this.addObjectType(
eventData.Muons,
this.getCompoundTrack,
'Muons',
false,
cuts,
);
}
if (eventData.Photons) {
const cuts = [
new Cut('phi', -pi, pi, 0.01),
new Cut('eta', -4, 4, 0.1),
new Cut('energy', 0, 10000),
new Cut('pT', 0, 50000),
];
this.addObjectType(
eventData.Photons,
this.getCompoundCluster,
'Photons',
false,
cuts,
);
}
if (eventData.Electrons) {
const cuts = [
new Cut('phi', -pi, pi, 0.01),
new Cut('eta', -4, 4, 0.1),
new Cut('energy', 0, 10000),
new Cut('pT', 0, 50000),
];
this.addObjectType(
eventData.Electrons,
this.getCompoundTrack,
'Electrons',
false,
cuts,
);
}
if (eventData.Vertices) {
const cuts = [new Cut('vertexType', 0, 5)];
const scaleVertices = (value: number) => {
this.graphicsLibrary
.getSceneManager()
.scaleChildObjects('Vertices', value);
};
const addVerticesOptions = this.addScaleOptions(
'verticesScale',
'Vertices Scale',
scaleVertices,
);
this.addObjectType(
eventData.Vertices,
PhoenixObjects.getVertex,
'Vertices',
false,
cuts,
addVerticesOptions,
);
}
if (eventData.MissingEnergy) {
const addMETSizeOption = (
typeFolder: GUI,
typeFolderPM: PhoenixMenuNode,
) => {
const scaleMET = (value: number) => {
this.graphicsLibrary
.getSceneManager()
.scaleChildObjects('MissingEnergy', value);
};
if (typeFolder) {
const sizeMenu = typeFolder
.add({ jetsScale: 100 }, 'jetsScale', 1, 200)
.name('Size (%)');
sizeMenu.onChange(scaleMET);
}
// Phoenix menu
if (typeFolderPM) {
typeFolderPM.addConfig({
type: 'slider',
label: 'Size (%)',
value: 100,
min: 1,
max: 200,
allowCustomValue: true,
onChange: scaleMET,
});
}
};
this.addObjectType(
eventData.MissingEnergy,
PhoenixObjects.getMissingEnergy,
'MissingEnergy',
false,
[],
addMETSizeOption,
);
}
}
/**
* Adds to the event display all collections of a given object type.
* @param object Contains all collections of a given type (Tracks, Jets, CaloClusters...).
* @param getObject Function that handles of reconstructing objects of the given type.
* @param typeName Label for naming the object type.
* @param concatonateObjs If true, don't process objects individually, but process as a group (e.g. for point hits). Default is false.
* @param cuts Filters that can be applied to the objects.
* @param extendEventDataTypeUI A callback to add more options to event data type UI folder.
*/
protected addObjectType(
object: any,
getObject: any,
typeName: string,
concatonateObjs: boolean = false,
cuts?: Cut[],
extendEventDataTypeUI?: (
typeFolder: GUI,
typeFolderPM: PhoenixMenuNode,
) => void,
) {
const objectGroup = this.graphicsLibrary.addEventDataTypeGroup(typeName);
const collectionsList: string[] = this.getObjectTypeCollections(object);
this.ui.addEventDataTypeFolder(typeName);
for (const collectionName of collectionsList) {
const newCuts = _.cloneDeep(cuts);
// Make a new array ^, otherwise we reuse the same cuts for each collection
const objectCollection = object[collectionName];
console.log(
`${typeName} collection ${collectionName} has ${objectCollection.length} constituents.`,
);
if (objectCollection.length == 0) {
console.log('Skipping');
continue;
}
this.addCollection(
objectCollection,
collectionName,
getObject,
typeName,
objectGroup,
concatonateObjs,
);
// collectionCuts is shallow copy
const collectionCuts = newCuts?.filter(
(cut) => cut.field in objectCollection[0],
);
this.ui.addCollection(typeName, collectionName, collectionCuts);
}
const eventDataTypeFolderDatGUI = this.ui
.getUIMenus()
.find((menu) => menu instanceof DatGUIMenuUI)
?.getEventDataTypeFolder(typeName) as GUI;
const eventDataTypeFolderPhoenixMenu = this.ui
.getUIMenus()
.find((menu) => menu instanceof PhoenixMenuUI)
?.getEventDataTypeFolder(typeName) as PhoenixMenuNode;
extendEventDataTypeUI?.(
eventDataTypeFolderDatGUI,
eventDataTypeFolderPhoenixMenu,
);
}
/**
* Adds to the event display all the objects inside a collection.
* @param objectCollection Contains the params for every object of the collection.
* @param collectionName Label to UNIQUELY identify the collection.
* @param getObject Handles reconstructing the objects of the collection.
* @param objectGroup Group containing the collections of the same object type.
* @param concatonateObjs If true, don't process objects individually, but process as a group (e.g. for point hits).
*/
private addCollection(
objectCollection: any,
collectionName: string,
getObject: (object: any, typeName: string) => Object3D,
typeName: string,
objectGroup: Group,
concatonateObjs: boolean,
) {
const collscene = new Group();
collscene.name = collectionName;
if (concatonateObjs) {
//in this case, we just pass the entire collection in
const object = getObject.bind(this)(objectCollection, typeName);
if (object) {
collscene.add(object);
}
} else {
for (const objectParams of objectCollection) {
const object = getObject.bind(this)(objectParams, typeName);
if (object) {
collscene.add(object);
}
}
}
objectGroup.add(collscene);
// console.log("-> Adding a threejs group called "+collscene.name+" with "+collscene.children.length+" children to the group called "+objectGroup.name);
}
/**
* Get collection names of a given object type.
* @param object Contains all collections of a given type (Tracks, Jets, CaloClusters etc.).
* @returns List of collection names of an object type (Tracks, Jets, CaloClusters etc.).
*/
private getObjectTypeCollections(object: any): string[] {
const collectionsList: string[] = [];
for (const collectionName in object) {
if (object[collectionName] !== null) {
collectionsList.push(collectionName);
}
}
return collectionsList;
}
/** Process the compound object of track type (e.g. Muon, Electron, ..) from the given parameters and get it as a group.
*/
protected getCompoundTrack(params: any, name = ''): Object3D {
return this.getCompound(params, name, true, false);
}
/** Process the compound object of cluster type (e.g. Photon, ..) from the given parameters and get it as a group.
*/
protected getCompoundCluster(params: any, name = ''): Object3D {
return this.getCompound(params, name, false, true);
}
/**
* Process the compound object (e.g. Muon, Electron, Photon) from the given parameters and get it as a group.
* FIXME. This is currently here and not in PhoenixObjects because we need to handle linked objects.
* @param params Parameters of the Muon.
* @returns Muon group containing Clusters and Tracks.
*/
protected getCompound(
params: any,
name = '',
objectIsTrack: boolean = false,
objectIsCluster: boolean = false,
): Object3D {
const scene = new Group();
// Try to find a linked cluster if possible, and draw this.
// If not (and if this makes sense), try to make a cluster from the compound object itself
if ('LinkedClusters' in params && params.LinkedClusters) {
for (const clusterID of params.LinkedClusters) {
const clusterColl = clusterID.split(':')[0];
const clusterIndex = clusterID.split(':')[1];
if (
clusterColl &&
clusterIndex &&
this.eventData.CaloClusters &&
this.eventData.CaloClusters[clusterColl]
) {
const clusterParams =
this.eventData.CaloClusters[clusterColl][clusterIndex];
if (clusterParams) {
const cluster = PhoenixObjects.getCluster(clusterParams);
cluster.name = name + ' Cluster';
scene.add(cluster);
}
}
}
} else if (objectIsCluster) {
const clusterParams = {
energy: params?.energy,
phi: params?.phi,
eta: params?.eta,
};
const cluster = PhoenixObjects.getCluster(clusterParams);
cluster.name = name + ' Cluster';
scene.add(cluster);
}
// Try to find a linked track if possible, and draw this.
// If not (and if this makes sense), try to extrapolate the compound object itself
if ('LinkedTracks' in params && params.LinkedTracks) {
for (const trackID of params.LinkedTracks) {
const trackColl = trackID.split(':')[0];
const trackIndex = trackID.split(':')[1];
if (
trackColl &&
trackIndex &&
this.eventData.Tracks &&
this.eventData.Tracks[trackColl]
) {
const trackParams = this.eventData.Tracks[trackColl][trackIndex];
if (trackParams) {
const track = PhoenixObjects.getTrack(trackParams);
if (track) {
track.name = name + ' Track';
scene.add(track);
} else {
console.log('WARNING: failed to get a track back.');
}
}
}
}
} else if (objectIsTrack) {
// We don't have links to any track, but maybe we can extrapolate something reasonable?
// ATLAS JiveXML have the following: energy, eta, phi, pt
const startPos = new Vector3(0, 0, 0);
const theta = CoordinateHelper.etaToTheta(params.eta);
const p = params.pt / Math.cos(Math.PI / 2 - theta);
let q = 0;
if ('pdgId' in params) {
q = params.pdgId > 0 ? 1 : -1;
}
// dparams = d0, z0, phi, theta, q/p
const trackparams = { dparams: [0, 0, params.phi, theta, q / p] };
const track = PhoenixObjects.getTrack(trackparams);
if (track) {
track.name = name + ' Track';
scene.add(track);
} else {
console.log('WARNING: failed to get a track back.');
}
}
// uuid for selection of compound obj from the collections info panel
params.uuid = scene.uuid;
scene.name = name;
// add to scene
return scene;
}
/**
* Get metadata associated to the event (experiment info, time, run, event...).
* @returns Metadata of the event.
*/
getEventMetadata(): any[] {
const metadata = [];
// Dividing event meta data into groups by keys and label
// For example, the first array group is for "Run / Event / LS"
const eventDataPropGroups = [
[
{ keys: ['runNumber', 'run number'], label: 'Run' },
{ keys: ['eventNumber', 'event number'], label: 'Event' },
{ keys: ['ls'], label: 'LS' },
{ keys: ['lumiBlock'], label: 'LumiBlock' },
],
[{ keys: ['time'], label: 'Data recorded' }],
];
const eventDataKeys = Object.keys(this.eventData);
// Iterating the group
for (const eventDataPropGroup of eventDataPropGroups) {
const combinedProps: { [key: string]: any } = {};
// Iterating the props inside a group
for (const eventDataProp of eventDataPropGroup) {
// Iterating each possible key of a prop
for (const eventDataPropKey of eventDataProp.keys) {
if (
eventDataKeys.includes(eventDataPropKey) &&
eventDataPropKey in this.eventData
) {
combinedProps[eventDataProp.label] =
this.eventData[eventDataPropKey];
break;
}
}
}
if (Object.keys(combinedProps).length > 0) {
// Joining and pushing the collected combined properties to the actual metadata
metadata.push({
label: Object.keys(combinedProps).join(' / '),
value: Object.values(combinedProps).join(' / '),
});
}
}
return metadata;
}
/**
* Add label of event object to the labels object.
* @param label Label to be saved.
* @param collection Collection the event object is a part of.
* @param indexInCollection Event object's index in collection.
* @returns A unique label ID string.
*/
public addLabelToEventObject(
label: string,
collection: string,
indexInCollection: number,
): string {
for (const eventDataType in this.eventData) {
if (this.eventData?.[eventDataType]?.[collection]) {
this.labelsObject[eventDataType] =
this.labelsObject[eventDataType] || {};
this.labelsObject[eventDataType][collection] =
this.labelsObject[eventDataType][collection] || {};
this.labelsObject[eventDataType][collection][indexInCollection] = label;
const tmp =
this.eventData[eventDataType][collection][indexInCollection];
tmp.labelText = label;
return getLabelTitle(eventDataType, collection, indexInCollection);
}
}
return 'Unknown';
}
/**
* Get the object containing labels.
* @returns The labels object.
*/
public getLabelsObject(): { [key: string]: any } {
return this.labelsObject;
}
/**
* Get function to add options to scale event object type by.
* @param configKey Key of the scale configuration option (for dat.GUI menu).
* @param configLabel Label of the scale configuration option.
* @param scaleFunction Function to scale the objects by.
* @returns Function which adds scale options to Phoenix menu and dat.GUI menu.
*/
public addScaleOptions(
configKey: string,
configLabel: string,
scaleFunction: (value: number) => void,
) {
return (typeFolder: GUI, typeFolderPM: PhoenixMenuNode) => {
// dat.GUI menu
if (typeFolder) {
const sizeMenu = typeFolder
.add({ [configKey]: 1 }, configKey, 0.001, 100)
.name(configLabel);
sizeMenu.onChange(scaleFunction);
}
// Phoenix menu
if (typeFolderPM) {
typeFolderPM.addConfig({
type: 'slider',
label: configLabel,
value: 1,
min: 0.001,
step: 0.01,
max: 100,
allowCustomValue: true,
onChange: scaleFunction,
});
}
};
}
}