File

src/event-display.ts

Description

Phoenix event display class for managing detector geometries and event data.

Index

Properties
Methods

Constructor

constructor(configuration?: Configuration)

Create the Phoenix event display and intitialize all the elements.

Parameters :
Name Type Optional Description
configuration Configuration Yes

Configuration used to customize different aspects.

Properties

Public configuration
Type : Configuration

Configuration for preset views and event data loader.

Private eventsData
Type : any

An object containing event data.

Private graphicsLibrary
Type : ThreeManager

Three manager for three.js operations.

Private infoLogger
Type : InfoLogger

Info logger for storing event display logs.

Private loadingManager
Type : LoadingManager

Loading manager for loadable resources

Private onDisplayedEventChange
Type : (function)[]
Default value : []

Array containing callbacks to be called when the displayed event changes.

Private onEventsChange
Type : (function)[]
Default value : []

Array containing callbacks to be called when events change.

Private stateManager
Type : StateManager

State manager for managing event display state.

Private ui
Type : UIManager

UI manager for UI menu.

Private urlOptionsManager
Type : URLOptionsManager

URL manager for managing options given through URL.

Methods

Public addLabelToObject
addLabelToObject(label: string, collection: string, indexInCollection: number, uuid: string)

Add label to a 3D object.

Parameters :
Name Type Optional Description
label string No

Label to add to the event object.

collection string No

Collection the event object is a part of.

indexInCollection number No

Event object's index in collection.

uuid string No

UUID of the three.js object.

Returns : void
Public allowSelection
allowSelection(selectedObject: literal type)

Initializes the object which will show information of the selected geometry/event data.

Parameters :
Name Type Optional Description
selectedObject literal type No

Object to display the data.

Returns : void
Public animateClippingWithCollision
animateClippingWithCollision(tweenDuration: number, onEnd?: () => void)

Animate the propagation and generation of event data using clipping planes after particle collison.

Parameters :
Name Type Optional Description
tweenDuration number No

Duration of the animation tween.

onEnd function Yes

Function to call when all animations have ended.

Returns : void
Public animateEventWithCollision
animateEventWithCollision(tweenDuration: number, onEnd?: () => void)

Animate the propagation and generation of event data with particle collison.

Parameters :
Name Type Optional Description
tweenDuration number No

Duration of the animation tween.

onEnd function Yes

Function to call when all animations have ended.

Returns : void
Public animatePreset
animatePreset(animationPreset: AnimationPreset, onEnd?: () => void)

Animate scene by animating camera through the scene and animating event collision. event collision animation options.

Parameters :
Name Type Optional Description
animationPreset AnimationPreset No

Preset for animation including positions to go through and event collision animation options.

onEnd function Yes

Function to call when the animation ends.

Returns : void
Public animateThroughEvent
animateThroughEvent(startPos: number[], tweenDuration: number, onAnimationEnd?: () => void)

Animate the camera through the event scene.

Parameters :
Name Type Optional Description
startPos number[] No

Start position of the translation animation.

tweenDuration number No

Duration of each tween in the translation animation.

onAnimationEnd function Yes

Callback when the last animation ends.

Returns : void
Public buildEventDataFromJSON
buildEventDataFromJSON(eventData: any)

Receives an object containing one event and builds the different collections of physics objects.

Parameters :
Name Type Optional Description
eventData any No

Object containing the event data.

Returns : void
Public buildGeometryFromParameters
buildGeometryFromParameters(parameters: any)

Build Geometry from thr passed parameters, where

Parameters :
Name Type Optional
parameters any No
Returns : void
Public disableHighlighting
disableHighlighting()

Disable highlighting of the objects.

Returns : void
Private enableEventDisplayConsole
enableEventDisplayConsole()

Enables calling specified event display methods in console.

Returns : void
Public enableHighlighting
enableHighlighting()

Enable highlighting of the objects.

Returns : void
Public enableKeyboardControls
enableKeyboardControls()

Enable keyboard controls for the event display.

Returns : void
Public enableSelecting
enableSelecting(enable: boolean)

Toggles the ability of selecting geometries/event data by clicking on the screen.

Parameters :
Name Type Optional Description
enable boolean No

Value to enable or disable the functionality.

Returns : void
Public endXR
endXR(xrSessionType: XRSessionType)

End VR and remove VR settings.

Parameters :
Name Type Optional Description
xrSessionType XRSessionType No

Type of the XR session. Either AR or VR.

Returns : void
Public exportPhoenixDisplay
exportPhoenixDisplay()

Exports scene as phoenix format, allowing to load it later and recover the saved configuration.

Returns : void
Public exportToOBJ
exportToOBJ()

Exports scene to OBJ file format.

Returns : void
Public fixOverlayView
fixOverlayView(fixed: boolean)

Fixes the camera position of the overlay view.

Parameters :
Name Type Optional Description
fixed boolean No

Whether the overlay view is to be fixed or not.

Returns : void
Public getActiveObjectId
getActiveObjectId()

Get the uuid of the currently selected object.

uuid of the currently selected object.

Public getCollection
getCollection(collectionName: string)

Get all the objects inside a collection.

Parameters :
Name Type Optional Description
collectionName string No

Key of the collection that will be retrieved.

Returns : any

Object containing all physics objects from the desired collection.

Public getCollections
getCollections()

Get the different collections for the current stored event.

Returns : string[]

List of strings, each representing a collection of the event displayed.

Public getEventMetadata
getEventMetadata()

Get metadata associated to the displayed event (experiment info, time, run, event...).

Returns : any[]

Metadata of the displayed event.

Public getInfoLogger
getInfoLogger()

Get the info logger containing event display logs.

Returns : InfoLogger

The info logger instance being used by the event display.

Public getLoadingManager
getLoadingManager()

Get the loading manager for managing loadable items.

Returns : LoadingManager

The loading manager.

Public getStateManager
getStateManager()

Get the state manager that manages event display's state.

Returns : StateManager

The state manager.

Public getThreeManager
getThreeManager()

Get the three manager responsible for three.js functions.

Returns : ThreeManager

The three.js manager.

Public getUIManager
getUIManager()

Get the UI manager responsible for UI related functions.

Returns : UIManager

The UI manager.

Public getURLOptionsManager
getURLOptionsManager()

Get the URL options manager that manages options given through URL.

Returns : URLOptionsManager

The URL options manager.

Public highlightObject
highlightObject(uuid: string, detector: boolean)

Highlight the object with the given uuid by giving it an outline.

Parameters :
Name Type Optional Default value Description
uuid string No

uuid of the object.

detector boolean No false

whether the function is for detector objects or event data.

Returns : void
Public init
init(configuration: Configuration)

Initialize all the Phoenix event display elements.

Parameters :
Name Type Optional Description
configuration Configuration No

Configuration used to customize different aspects.

Returns : void
Public initXR
initXR(xrSessionType: XRSessionType, onSessionEnded?: () => void)

Initialize XR.

Parameters :
Name Type Optional Description
xrSessionType XRSessionType No

Type of the XR session. Either AR or VR.

onSessionEnded function Yes

Callback when the XR session ends.

Returns : void
Public listenToDisplayedEventChange
listenToDisplayedEventChange(callback: (event: any) => void)

Add a callback to onDisplayedEventChange array to call the callback on changes to the displayed event.

Parameters :
Name Type Optional Description
callback function No

Callback to be added to the onDisplayedEventChange array.

Returns : void
Public listenToLoadedEventsChange
listenToLoadedEventsChange(callback: (events: any) => void)

Add a callback to onEventsChange array to call the callback on changes to the events.

Parameters :
Name Type Optional Description
callback function No

Callback to be added to the onEventsChange array.

Returns : void
Public loadEvent
loadEvent(eventKey: any)

Receives a string representing the key of an event and loads the event associated with that key.

Parameters :
Name Type Optional Description
eventKey any No

String that represents the event in the eventsData object.

Returns : void
Public Async loadGLTFGeometry
loadGLTFGeometry(url: any, name: string, menuNodeName: string, scale: number, initiallyVisible: boolean)

Loads a GLTF (.gltf) scene/geometry from the given URL. and adds it to the dat.GUI menu.

Parameters :
Name Type Optional Default value Description
url any No

URL to the GLTF (.gltf) file.

name string No

Name of the loaded scene/geometry.

menuNodeName string No ''

Name of the node in Phoenix menu to add the geometry to. Use > as a separator for specifying the hierarchy for sub-folders.

scale number No 1.0

Scale of the geometry.

initiallyVisible boolean No true

Whether the geometry is initially visible or not. Default true.

Returns : Promise<void>

Promise for loading the geometry.

Public Async loadJSONGeometry
loadJSONGeometry(json: string | literal type, name: string, menuNodeName?: string, scale?: number, doubleSided?: boolean, initiallyVisible: boolean)

Loads geometries from JSON.

Parameters :
Name Type Optional Default value Description
json string | literal type No

JSON or URL to JSON file of the geometry.

name string No

Name of the geometry or group of geometries.

menuNodeName string Yes

Name of the node in Phoenix menu to add the geometry to. Use > as a separator for specifying the hierarchy for sub-folders.

scale number Yes

Scale of the geometry.

doubleSided boolean Yes

Renders both sides of the material.

initiallyVisible boolean No true

Whether the geometry is initially visible or not. Default true.

Returns : Promise<void>

Promise for loading the geometry.

Public Async loadOBJGeometry
loadOBJGeometry(filename: string, name: string, color: any, menuNodeName: string, doubleSided: boolean, initiallyVisible: boolean, setFlat: boolean)

Loads an OBJ (.obj) geometry from the given filename and adds it to the dat.GUI menu.

Parameters :
Name Type Optional Default value Description
filename string No

Path to the geometry.

name string No

Name given to the geometry.

color any No

Color to initialize the geometry.

menuNodeName string No

Name of the node in Phoenix menu to add the geometry to. Use > as a separator for specifying the hierarchy for sub-folders.

doubleSided boolean No

If true, render both sides of the material.

initiallyVisible boolean No true

Whether the geometry is initially visible or not. Default true.

setFlat boolean No true

Whether object should be flat-shaded or not. Default true.

Returns : Promise<void>

Promise for loading the geometry.

Public Async loadRootGeometry
loadRootGeometry(url: string, objectName: string, name: string, menuNodeName?: string, scale?: number, doubleSided?: boolean, initiallyVisible: boolean)

Load ROOT geometry from JSRoot.

Parameters :
Name Type Optional Default value Description
url string No

URL of the JSRoot file.

objectName string No

Name of the object inside the ".root" file.

name string No

Name of the geometry.

menuNodeName string Yes

Name of the node in Phoenix menu to add the geometry to. Use > as a separator for specifying the hierarchy for sub-folders.

scale number Yes

Scale of the geometry.

doubleSided boolean Yes

Renders both sides of the material.

initiallyVisible boolean No true

Whether the geometry is initially visible or not. Default true.

Returns : any
Public Async loadRootJSONGeometry
loadRootJSONGeometry(url: string, name: string, menuNodeName?: string, scale?: number, doubleSided?: boolean, initiallyVisible: boolean)

Load JSON geometry from JSRoot.

Parameters :
Name Type Optional Default value Description
url string No

URL of the JSRoot geometry file.

name string No

Name of the geometry.

menuNodeName string Yes

Name of the node in Phoenix menu to add the geometry to. Use > as a separator for specifying the hierarchy for sub-folders.

scale number Yes

Scale of the geometry.

doubleSided boolean Yes

Renders both sides of the material.

initiallyVisible boolean No true

Whether the geometry is initially visible or not. Default true.

Returns : any
Private loadSceneConfiguration
loadSceneConfiguration(sceneConfiguration: literal type)

Processes event data and geometry for Loading the scene from Phoenix file format (.phnx).

Parameters :
Name Type Optional Description
sceneConfiguration literal type No

Scene configuration containingevent data and detector geometry.

Returns : void
Public lookAtObject
lookAtObject(uuid: string, detector: boolean)

Move the camera to look at the object with the given uuid and highlight it.

Parameters :
Name Type Optional Default value Description
uuid string No

uuid of the object.

detector boolean No false

whether the function is for detector objects or event data.

Returns : void
Public Async parseGLTFGeometry
parseGLTFGeometry(file: File)

Parses and loads a geometry in GLTF (.gltf or .glb) format. also supports zip files of the above

Parameters :
Name Type Optional Description
file File No

Geometry file in GLTF (.gltf or .glb) format.

Returns : Promise<void>

Promise for loading the geometry.

Public parseOBJGeometry
parseOBJGeometry(content: string, name: string, menuNodeName?: string, initiallyVisible: boolean)

Parses and loads an OBJ geometry from the given content and adds it to the dat.GUI menu.

Parameters :
Name Type Optional Default value Description
content string No

Content of the OBJ geometry.

name string No

Name given to the geometry.

menuNodeName string Yes

Name of the node in Phoenix menu to add the geometry to. Use > as a separator for specifying the hierarchy for sub-folders.

initiallyVisible boolean No true

Whether the geometry is initially visible or not. Default true.

Returns : void
Public Async parsePhoenixDisplay
parsePhoenixDisplay(input: any)

Parse and load an event from the Phoenix file format (.phnx). and other configuration.

Parameters :
Name Type Optional Description
input any No

Content containing the JSON with event data and other configuration.

Returns : Promise<void>

Promise for loading the geometry.

Public parsePhoenixEvents
parsePhoenixEvents(eventsData: any)

Receives an object containing all the eventKeys and saves it. Then it loads by default the first event.

Parameters :
Name Type Optional Description
eventsData any No

Object containing the event data.

Returns : string[]

Array of strings containing the keys of the eventsData object.

Public resetLabels
resetLabels()

Reset/remove all labels.

Returns : void
Public setOverlayRenderer
setOverlayRenderer(overlayCanvas: HTMLCanvasElement)

Sets the renderer to be used to render the event display on the overlayed canvas.

Parameters :
Name Type Optional Description
overlayCanvas HTMLCanvasElement No

An HTML canvas on which the overlay renderer is to be set.

Returns : void
Public zoomTo
zoomTo(zoomFactor: number, zoomTime: number)

Zoom all the cameras by a specific zoom factor. The factor may either be greater (zoom in) or smaller (zoom out) than 1.

Parameters :
Name Type Optional Description
zoomFactor number No

The factor to zoom by.

zoomTime number No

The time it takes for a zoom animation to complete.

Returns : void
import { httpRequest, openFile } from 'jsroot';
import { settings as jsrootSettings } from 'jsroot';
import { build } from 'jsroot/geom';
import { ThreeManager } from './managers/three-manager/index';
import { UIManager } from './managers/ui-manager/index';
import { InfoLogger } from './helpers/info-logger';
import type { Configuration } from './lib/types/configuration';
import { StateManager } from './managers/state-manager';
import { LoadingManager } from './managers/loading-manager';
import { URLOptionsManager } from './managers/url-options-manager';
import { ActiveVariable } from './helpers/active-variable';
import type { AnimationPreset } from './managers/three-manager/animations-manager';
import { XRSessionType } from './managers/three-manager/xr/xr-manager';
import { getLabelTitle } from './helpers/labels';
import { PhoenixLoader } from './loaders/phoenix-loader';

declare global {
  /**
   * Window interface for adding objects to the window object.
   */
  interface Window {
    /** EventDisplay object containing event display related functions. */
    EventDisplay: any;
  }
}

/**
 * Phoenix event display class for managing detector geometries and event data.
 */
export class EventDisplay {
  /** Configuration for preset views and event data loader. */
  public configuration: Configuration;
  /** An object containing event data. */
  private eventsData: any;
  /** Array containing callbacks to be called when events change. */
  private onEventsChange: ((events: any) => void)[] = [];
  /** Array containing callbacks to be called when the displayed event changes. */
  private onDisplayedEventChange: ((nowDisplayingEvent: any) => void)[] = [];
  /** Three manager for three.js operations. */
  private graphicsLibrary: ThreeManager;
  /** Info logger for storing event display logs. */
  private infoLogger: InfoLogger;
  /** UI manager for UI menu. */
  private ui: UIManager;
  /** Loading manager for loadable resources */
  private loadingManager: LoadingManager;
  /** State manager for managing event display state. */
  private stateManager: StateManager;
  /** URL manager for managing options given through URL. */
  private urlOptionsManager: URLOptionsManager;

  /**
   * Create the Phoenix event display and intitialize all the elements.
   * @param configuration Configuration used to customize different aspects.
   */
  constructor(configuration?: Configuration) {
    this.loadingManager = new LoadingManager();
    this.infoLogger = new InfoLogger();
    this.graphicsLibrary = new ThreeManager(this.infoLogger);
    this.ui = new UIManager(this.graphicsLibrary);
    if (configuration) {
      this.init(configuration);
    }
  }

  /**
   * Initialize all the Phoenix event display elements.
   * @param configuration Configuration used to customize different aspects.
   */
  public init(configuration: Configuration) {
    this.configuration = configuration;

    // Initialize the three manager with configuration
    this.graphicsLibrary.init(configuration);
    // Initialize the UI with configuration
    this.ui.init(configuration);
    // Set up for the state manager
    this.getStateManager().setEventDisplay(this);

    // Animate loop
    const uiLoop = () => {
      this.ui.updateUI();
    };
    this.graphicsLibrary.setAnimationLoop(uiLoop);

    // Process and apply URL options
    this.urlOptionsManager = new URLOptionsManager(this, configuration);
    if (configuration.allowUrlOptions !== false) {
      this.urlOptionsManager.applyOptions();
    }

    // Allow adding elements through console
    this.enableEventDisplayConsole();
    // Allow keyboard controls
    this.enableKeyboardControls();
  }

  /**
   * Initialize XR.
   * @param xrSessionType Type of the XR session. Either AR or VR.
   * @param onSessionEnded Callback when the XR session ends.
   */
  public initXR(xrSessionType: XRSessionType, onSessionEnded?: () => void) {
    this.graphicsLibrary.initXRSession(xrSessionType, onSessionEnded);
  }

  /**
   * End VR and remove VR settings.
   * @param xrSessionType Type of the XR session. Either AR or VR.
   */
  public endXR(xrSessionType: XRSessionType) {
    this.graphicsLibrary.endXRSession(xrSessionType);
  }

  /**
   * Receives an object containing all the eventKeys and saves it.
   * Then it loads by default the first event.
   * @param eventsData Object containing the event data.
   * @returns Array of strings containing the keys of the eventsData object.
   */
  public parsePhoenixEvents(eventsData: any): string[] {
    this.eventsData = eventsData;
    if (typeof this.configuration.eventDataLoader === 'undefined') {
      this.configuration.eventDataLoader = new PhoenixLoader();
    }
    const eventKeys =
      this.configuration.eventDataLoader.getEventsList(eventsData);
    this.loadEvent(eventKeys[0]);
    this.onEventsChange.forEach((callback) => callback(eventKeys));

    return eventKeys;
  }

  /**
   * Receives an object containing one event and builds the different collections
   * of physics objects.
   * @param eventData Object containing the event data.
   */
  public buildEventDataFromJSON(eventData: any) {
    // Reset labels
    this.resetLabels();
    // Creating UI folder
    this.ui.addEventDataFolder();
    this.ui.addLabelsFolder();
    // Clearing existing event data
    this.graphicsLibrary.clearEventData();
    // Build data and add to scene
    if (this.configuration.eventDataLoader) {
      this.configuration.eventDataLoader.buildEventData(
        eventData,
        this.graphicsLibrary,
        this.ui,
        this.infoLogger,
      );
    }
    this.onDisplayedEventChange.forEach((callback) => callback(eventData));
    // Reload the event data state in Phoenix menu
    this.ui.loadEventFolderPhoenixMenuState();
  }

  /**
   * Receives a string representing the key of an event and loads
   * the event associated with that key.
   * @param eventKey String that represents the event in the eventsData object.
   */
  public loadEvent(eventKey: any) {
    const event = this.eventsData[eventKey];

    if (event) {
      this.buildEventDataFromJSON(event);
    }
  }

  /**
   * Get the three manager responsible for three.js functions.
   * @returns The three.js manager.
   */
  public getThreeManager() {
    return this.graphicsLibrary;
  }

  /**
   * Get the UI manager responsible for UI related functions.
   * @returns The UI manager.
   */
  public getUIManager() {
    return this.ui;
  }

  /**
   * Get the info logger containing event display logs.
   * @returns The info logger instance being used by the event display.
   */
  public getInfoLogger() {
    return this.infoLogger;
  }

  /**
   * Get the loading manager for managing loadable items.
   * @returns The loading manager.
   */
  public getLoadingManager() {
    return this.loadingManager;
  }

  /**
   * Get the state manager that manages event display's state.
   * @returns The state manager.
   */
  public getStateManager() {
    if (!this.stateManager) {
      this.stateManager = new StateManager();
    }

    return this.stateManager;
  }

  /**
   * Get the URL options manager that manages options given through URL.
   * @returns The URL options manager.
   */
  public getURLOptionsManager() {
    return this.urlOptionsManager;
  }

  // **********************
  // * LOADING GEOMETRIES *
  // **********************

  /**
   * Loads an OBJ (.obj) geometry from the given filename
   * and adds it to the dat.GUI menu.
   * @param filename Path to the geometry.
   * @param name Name given to the geometry.
   * @param color Color to initialize the geometry.
   * @param menuNodeName Name of the node in Phoenix menu to add the geometry to. Use >  as a separator for specifying the hierarchy for sub-folders.
   * @param doubleSided If true, render both sides of the material.
   * @param initiallyVisible Whether the geometry is initially visible or not. Default `true`.
   * @param setFlat Whether object should be flat-shaded or not. Default `true`.
   * @returns Promise for loading the geometry.
   */
  public async loadOBJGeometry(
    filename: string,
    name: string,
    color: any,
    menuNodeName: string,
    doubleSided: boolean,
    initiallyVisible: boolean = true,
    setFlat: boolean = true,
  ): Promise<void> {
    this.loadingManager.addLoadableItem(`obj_geom_${name}`);

    const { object } = await this.graphicsLibrary.loadOBJGeometry(
      filename,
      name,
      color,
      doubleSided,
      initiallyVisible,
      setFlat,
    );
    this.ui.addGeometry(object, menuNodeName);

    this.loadingManager.itemLoaded(`obj_geom_${name}`);
    this.infoLogger.add(name, 'Loaded OBJ geometry');
  }

  /**
   * Parses and loads an OBJ geometry from the given content
   * and adds it to the dat.GUI menu.
   * @param content Content of the OBJ geometry.
   * @param name Name given to the geometry.
   * @param menuNodeName Name of the node in Phoenix menu to add the geometry to. Use >  as a separator for specifying the hierarchy for sub-folders.
   * @param initiallyVisible Whether the geometry is initially visible or not. Default `true`.
   */
  public parseOBJGeometry(
    content: string,
    name: string,
    menuNodeName?: string,
    initiallyVisible: boolean = true,
  ) {
    this.loadingManager.addLoadableItem(`parse_obj_${name}`);
    const { object } = this.graphicsLibrary.parseOBJGeometry(
      content,
      name,
      initiallyVisible,
    );
    this.ui.addGeometry(object, menuNodeName);
    this.loadingManager.itemLoaded(`parse_obj_${name}`);
  }

  /**
   * Exports scene to OBJ file format.
   */
  public exportToOBJ() {
    this.graphicsLibrary.exportSceneToOBJ();
    this.infoLogger.add('Exported scene to OBJ');
  }

  /**
   * Parse and load an event from the Phoenix file format (.phnx).
   * @param input Content containing the JSON with event data
   * and other configuration.
   * @returns Promise for loading the geometry.
   */
  public async parsePhoenixDisplay(input: any): Promise<void> {
    const phoenixScene = JSON.parse(input);

    if (phoenixScene.sceneConfiguration && phoenixScene.scene) {
      // Creating UI folder
      this.ui.addEventDataFolder();
      this.ui.addLabelsFolder();
      // Clearing existing event data
      this.graphicsLibrary.clearEventData();
      // Add to scene
      this.loadSceneConfiguration(phoenixScene.sceneConfiguration);

      this.loadingManager.addLoadableItem(`parse_phnx_${name}`);
      await this.graphicsLibrary.parsePhnxScene(phoenixScene.scene);
      this.loadingManager.itemLoaded(`parse_phnx_${name}`);
    }
  }

  /**
   * Exports scene as phoenix format, allowing to load it later and recover the saved configuration.
   */
  public exportPhoenixDisplay() {
    this.graphicsLibrary.exportPhoenixScene();
  }

  /**
   * Parses and loads a geometry in GLTF (.gltf or .glb) format.
   * also supports zip files of the above
   * @param file Geometry file in GLTF (.gltf or .glb) format.
   * @returns Promise for loading the geometry.
   */
  public async parseGLTFGeometry(file: File): Promise<void> {
    const name = file.name.split('/').pop();
    this.loadingManager.addLoadableItem(`parse_gltf_${name}`);

    const allGeometriesUIParameters =
      await this.graphicsLibrary.parseGLTFGeometry(file);
    for (const { object, menuNodeName } of allGeometriesUIParameters) {
      this.ui.addGeometry(object, menuNodeName);
    }

    this.loadingManager.itemLoaded(`parse_gltf_${name}`);
  }

  /**
   * Loads a GLTF (.gltf) scene/geometry from the given URL.
   * and adds it to the dat.GUI menu.
   * @param url URL to the GLTF (.gltf) file.
   * @param name Name of the loaded scene/geometry.
   * @param menuNodeName Name of the node in Phoenix menu to add the geometry to. Use >  as a separator for specifying the hierarchy for sub-folders.
   * @param scale Scale of the geometry.
   * @param initiallyVisible Whether the geometry is initially visible or not. Default `true`.
   * @returns Promise for loading the geometry.
   */
  public async loadGLTFGeometry(
    url: any,
    name: string,
    menuNodeName: string = '',
    scale: number = 1.0,
    initiallyVisible: boolean = true,
  ): Promise<void> {
    this.loadingManager.addLoadableItem(`gltf_geom_${name}`);

    const allGeometriesUIParameters =
      await this.graphicsLibrary.loadGLTFGeometry(
        url,
        name,
        menuNodeName,
        scale,
        initiallyVisible,
      );
    for (const { object, menuNodeName } of allGeometriesUIParameters) {
      this.ui.addGeometry(object, menuNodeName);
    }

    this.loadingManager.itemLoaded(`gltf_geom_${name}`);
  }

  /**
   * Loads geometries from JSON.
   * @param json JSON or URL to JSON file of the geometry.
   * @param name Name of the geometry or group of geometries.
   * @param menuNodeName Name of the node in Phoenix menu to add the geometry to. Use >  as a separator for specifying the hierarchy for sub-folders.
   * @param scale Scale of the geometry.
   * @param doubleSided Renders both sides of the material.
   * @param initiallyVisible Whether the geometry is initially visible or not. Default `true`.
   * @returns Promise for loading the geometry.
   */
  public async loadJSONGeometry(
    json: string | { [key: string]: any },
    name: string,
    menuNodeName?: string,
    scale?: number,
    doubleSided?: boolean,
    initiallyVisible: boolean = true,
  ): Promise<void> {
    this.loadingManager.addLoadableItem(`json_geom_${name}`);

    const { object } = await this.graphicsLibrary.loadJSONGeometry(
      json,
      name,
      scale,
      doubleSided,
      initiallyVisible,
    );
    this.ui.addGeometry(object, menuNodeName);

    this.loadingManager.itemLoaded(`json_geom_${name}`);
    this.infoLogger.add(name, 'Loaded JSON geometry');
  }

  /**
   * Load JSON geometry from JSRoot.
   * @param url URL of the JSRoot geometry file.
   * @param name Name of the geometry.
   * @param menuNodeName Name of the node in Phoenix menu to add the geometry to. Use >  as a separator for specifying the hierarchy for sub-folders.
   * @param scale Scale of the geometry.
   * @param doubleSided Renders both sides of the material.
   * @param initiallyVisible Whether the geometry is initially visible or not. Default `true`.
   */
  public async loadRootJSONGeometry(
    url: string,
    name: string,
    menuNodeName?: string,
    scale?: number,
    doubleSided?: boolean,
    initiallyVisible: boolean = true,
  ) {
    this.loadingManager.addLoadableItem('root_json_geom');

    const object = await httpRequest(url, 'object');
    await this.loadJSONGeometry(
      build(object, { dflt_colors: true }).toJSON(),
      name,
      menuNodeName,
      scale,
      doubleSided,
      initiallyVisible,
    );

    this.loadingManager.itemLoaded('root_json_geom');
  }

  /**
   * Load ROOT geometry from JSRoot.
   * @param url URL of the JSRoot file.
   * @param objectName Name of the object inside the ".root" file.
   * @param name Name of the geometry.
   * @param menuNodeName Name of the node in Phoenix menu to add the geometry to. Use >  as a separator for specifying the hierarchy for sub-folders.
   * @param scale Scale of the geometry.
   * @param doubleSided Renders both sides of the material.
   * @param initiallyVisible Whether the geometry is initially visible or not. Default `true`.
   */
  public async loadRootGeometry(
    url: string,
    objectName: string,
    name: string,
    menuNodeName?: string,
    scale?: number,
    doubleSided?: boolean,
    initiallyVisible: boolean = true,
  ) {
    this.loadingManager.addLoadableItem('root_geom');
    // See https://github.com/root-project/jsroot/blob/19ce116b68701ab45e0a092c673119bf97ede0c2/modules/core.mjs#L241.
    jsrootSettings.UseStamp = false;

    const file = await openFile(url);
    const obj = await file.readObject(objectName);

    await this.loadJSONGeometry(
      build(obj, { dflt_colors: true }).toJSON(),
      name,
      menuNodeName,
      scale,
      doubleSided,
      initiallyVisible,
    );

    this.loadingManager.itemLoaded('root_geom');
  }

  /**
   * Build Geometry from thr passed parameters, where
   * @param parameters
   */
  public buildGeometryFromParameters(parameters: any): void {
    this.graphicsLibrary.addGeometryFromParameters(parameters);
  }

  /**
   * Zoom all the cameras by a specific zoom factor.
   * The factor may either be greater (zoom in) or smaller (zoom out) than 1.
   * @param zoomFactor The factor to zoom by.
   * @param zoomTime The time it takes for a zoom animation to complete.
   */
  public zoomTo(zoomFactor: number, zoomTime: number) {
    this.graphicsLibrary.zoomTo(zoomFactor, zoomTime);
  }

  /**
   * Processes event data and geometry for Loading the scene
   * from Phoenix file format (.phnx).
   * @param sceneConfiguration Scene configuration containingevent data and detector geometry.
   */
  private loadSceneConfiguration(sceneConfiguration: {
    eventData: { [key: string]: any };
    geometries: any[];
  }) {
    for (const objectType of Object.keys(sceneConfiguration.eventData)) {
      this.ui.addEventDataTypeFolder(objectType);

      const collections = sceneConfiguration.eventData[objectType];
      for (const collection of collections) {
        this.ui.addCollection(objectType, collection);
      }
    }

    for (const geom of sceneConfiguration.geometries) {
      this.ui.addGeometry(geom, '#ffffff');
    }
  }

  /**
   * Get all the objects inside a collection.
   * @param collectionName Key of the collection that will be retrieved.
   * @returns Object containing all physics objects from the desired collection.
   */
  public getCollection(collectionName: string) {
    if (this.configuration.eventDataLoader) {
      return this.configuration.eventDataLoader.getCollection(collectionName);
    }
    return {};
  }

  /**
   * Get the different collections for the current stored event.
   * @returns List of strings, each representing a collection of the event displayed.
   */
  public getCollections(): string[] {
    if (this.configuration.eventDataLoader) {
      return this.configuration.eventDataLoader.getCollections();
    }
    return [];
  }

  /**
   * Add a callback to onDisplayedEventChange array to call
   * the callback on changes to the displayed event.
   * @param callback Callback to be added to the onDisplayedEventChange array.
   */
  public listenToDisplayedEventChange(callback: (event: any) => any) {
    this.onDisplayedEventChange.push(callback);
  }

  /**
   * Add a callback to onEventsChange array to call
   * the callback on changes to the events.
   * @param callback Callback to be added to the onEventsChange array.
   */
  public listenToLoadedEventsChange(callback: (events: any) => any) {
    this.onEventsChange.push(callback);
  }

  /**
   * Get metadata associated to the displayed event (experiment info, time, run, event...).
   * @returns Metadata of the displayed event.
   */
  public getEventMetadata(): any[] {
    if (this.configuration.eventDataLoader) {
      return this.configuration.eventDataLoader.getEventMetadata();
    }
    return [];
  }

  /**
   * Enables calling specified event display methods in console.
   */
  private enableEventDisplayConsole() {
    // Defining an EventDisplay object in window to access methods through console
    window.EventDisplay = {
      loadGLTFGeometry: (sceneUrl: string, name: string) => {
        this.loadGLTFGeometry(sceneUrl, name);
      },
      loadOBJGeometry: (
        filename: string,
        name: string,
        colour: any,
        menuNodeName: string,
        doubleSided: boolean,
      ) => {
        this.loadOBJGeometry(filename, name, colour, menuNodeName, doubleSided);
      },
      loadJSONGeometry: (
        json: string | { [key: string]: any },
        name: string,
        menuNodeName: string,
        scale?: number,
        doubleSided?: boolean,
        initiallyVisible: boolean = true,
      ) => {
        this.loadJSONGeometry(
          json,
          name,
          menuNodeName,
          scale,
          doubleSided,
          initiallyVisible,
        );
      },
      buildGeometryFromParameters: (parameters: { [key: string]: any }) =>
        this.buildGeometryFromParameters(parameters),
      scene: this.getThreeManager().getSceneManager().getScene(),
    };
  }

  /**
   * Sets the renderer to be used to render the event display on the overlayed canvas.
   * @param overlayCanvas An HTML canvas on which the overlay renderer is to be set.
   */
  public setOverlayRenderer(overlayCanvas: HTMLCanvasElement) {
    this.graphicsLibrary.setOverlayRenderer(overlayCanvas);
  }

  /**
   * Initializes the object which will show information of the selected geometry/event data.
   * @param selectedObject Object to display the data.
   */
  public allowSelection(selectedObject: { name: string; attributes: any[] }) {
    this.graphicsLibrary.setSelectedObjectDisplay(selectedObject);
  }

  /**
   * Toggles the ability of selecting geometries/event data by clicking on the screen.
   * @param enable Value to enable or disable the functionality.
   */
  public enableSelecting(enable: boolean) {
    this.graphicsLibrary.enableSelecting(enable);
  }

  /**
   * Fixes the camera position of the overlay view.
   * @param fixed Whether the overlay view is to be fixed or not.
   */
  public fixOverlayView(fixed: boolean) {
    this.graphicsLibrary.fixOverlayView(fixed);
  }

  /**
   * Get the uuid of the currently selected object.
   * @returns uuid of the currently selected object.
   */
  public getActiveObjectId(): ActiveVariable<string> {
    return this.graphicsLibrary.getActiveObjectId();
  }

  /**
   * Move the camera to look at the object with the given uuid
   * and highlight it.
   * @param uuid uuid of the object.
   * @param detector whether the function is for detector objects or event data.
   */
  public lookAtObject(uuid: string, detector: boolean = false) {
    if (detector == true) {
      this.graphicsLibrary.lookAtObject(uuid, true);
      this.graphicsLibrary.highlightObject(uuid, true);
    } else {
      this.graphicsLibrary.lookAtObject(uuid);
      this.graphicsLibrary.highlightObject(uuid);
    }
  }

  /**
   * Highlight the object with the given uuid by giving it an outline.
   * @param uuid uuid of the object.
   * @param detector whether the function is for detector objects or event data.
   */
  public highlightObject(uuid: string, detector: boolean = false) {
    if (detector == true) {
      this.graphicsLibrary.highlightObject(uuid, true);
    } else {
      this.graphicsLibrary.highlightObject(uuid, false);
    }
  }

  /**
   * Enable highlighting of the objects.
   */
  public enableHighlighting() {
    this.graphicsLibrary.enableHighlighting();
  }

  /**
   * Disable highlighting of the objects.
   */
  public disableHighlighting() {
    this.graphicsLibrary.disableHighlighting();
  }

  /**
   * Enable keyboard controls for the event display.
   */
  public enableKeyboardControls() {
    this.ui.enableKeyboardControls();
    this.graphicsLibrary.enableKeyboardControls();
  }

  /**
   * Animate the camera through the event scene.
   * @param startPos Start position of the translation animation.
   * @param tweenDuration Duration of each tween in the translation animation.
   * @param onAnimationEnd Callback when the last animation ends.
   */
  public animateThroughEvent(
    startPos: number[],
    tweenDuration: number,
    onAnimationEnd?: () => void,
  ) {
    this.graphicsLibrary.animateThroughEvent(
      startPos,
      tweenDuration,
      onAnimationEnd,
    );
  }

  /**
   * Animate scene by animating camera through the scene and animating event collision.
   * @param animationPreset Preset for animation including positions to go through and
   * event collision animation options.
   * @param onEnd Function to call when the animation ends.
   */
  public animatePreset(animationPreset: AnimationPreset, onEnd?: () => void) {
    this.graphicsLibrary.animatePreset(animationPreset, onEnd);
  }

  /**
   * Animate the propagation and generation of event data with particle collison.
   * @param tweenDuration Duration of the animation tween.
   * @param onEnd Function to call when all animations have ended.
   */
  public animateEventWithCollision(tweenDuration: number, onEnd?: () => void) {
    this.graphicsLibrary.animateEventWithCollision(tweenDuration, onEnd);
  }

  /**
   * Animate the propagation and generation of event data
   * using clipping planes after particle collison.
   * @param tweenDuration Duration of the animation tween.
   * @param onEnd Function to call when all animations have ended.
   */
  public animateClippingWithCollision(
    tweenDuration: number,
    onEnd?: () => void,
  ) {
    this.graphicsLibrary.animateClippingWithCollision(tweenDuration, onEnd);
  }

  /**
   * Add label to a 3D object.
   * @param label Label to add to the event object.
   * @param collection Collection the event object is a part of.
   * @param indexInCollection Event object's index in collection.
   * @param uuid UUID of the three.js object.
   */
  public addLabelToObject(
    label: string,
    collection: string,
    indexInCollection: number,
    uuid: string,
  ) {
    if (!this.configuration.eventDataLoader) {
      return;
    }

    const labelId = this.configuration.eventDataLoader.addLabelToEventObject(
      label,
      collection,
      indexInCollection,
    );

    // Remove the label if the string is empty
    if (!label) {
      this.ui.removeLabel(labelId, true);
      return;
    }

    this.ui.addLabel(labelId);
    this.graphicsLibrary.addLabelToObject(label, uuid, labelId);
  }

  /**
   * Reset/remove all labels.
   */
  public resetLabels() {
    // labelsObject[EventDataType][Collection][Index]
    if (!this.configuration.eventDataLoader) {
      return;
    }
    const labelsObject = this.configuration.eventDataLoader.getLabelsObject();
    for (const eventDataType in labelsObject) {
      for (const collection in labelsObject[eventDataType]) {
        for (const index in labelsObject[eventDataType][collection]) {
          const labelId = getLabelTitle(eventDataType, collection, index);
          this.ui.removeLabel(labelId, true);

          delete labelsObject[eventDataType][collection][index];
        }
      }
    }
  }
}

results matching ""

    No results matching ""