src/managers/three-manager/controls-manager.ts
Manager for managing event display controls.
Properties |
|
Methods |
|
constructor(rendererManager: RendererManager, defaultView: number[])
|
||||||||||||
Constructor for setting up all the controls.
Parameters :
|
Private activeControls |
Type : OrbitControls
|
Currently active orbit controls. |
Private controls |
Type : OrbitControls[]
|
All orbit controls. |
Private mainControls |
Type : OrbitControls
|
The main orbit controls. |
Private orthographicControls |
Type : OrbitControls
|
Orbit controls for the orthographic view. |
Private overlayControls |
Type : OrbitControls
|
Orbit controls for overlay view. |
Private perspectiveControls |
Type : OrbitControls
|
Orbit controls for the perspective view. |
Public addControls | ||||||||
addControls(controls: OrbitControls)
|
||||||||
Add orbit controls to the controls list.
Parameters :
Returns :
void
|
Private containsObject | ||||||||||||
containsObject(obj: OrbitControls, list: OrbitControls[])
|
||||||||||||
Check if the list of orbit controls contains a specific orbit controls.
Parameters :
Returns :
boolean
If the list contains the orbit controls. |
getActiveCamera |
getActiveCamera()
|
Get the currently active camera.
Returns :
Camera
Currently active camera. |
getActiveControls |
getActiveControls()
|
Get currently active orbit controls.
Returns :
OrbitControls
Currently active orbit controls. |
getAllCameras |
getAllCameras()
|
Get the main and overlay cameras.
Returns :
Camera[]
An array containing the main and overlay cameras. |
getMainCamera |
getMainCamera()
|
Get the main camera.
Returns :
Camera
Main camera. |
getMainControls |
getMainControls()
|
Get the main orbit controls.
Returns :
OrbitControls
Main orbit controls. |
Public getObjectPosition | ||||||||||||
getObjectPosition(uuid: string, objectsGroup: Object3D)
|
||||||||||||
Get position of object from UUID.
Parameters :
Returns :
Vector3
Position of the 3D object. |
getOverlayCamera |
getOverlayCamera()
|
Get the camera for overlay.
Returns :
Camera
The camera for overlay. |
getOverlayControls |
getOverlayControls()
|
Get orbit controls for overlay.
Returns :
OrbitControls
Orbit controls for overlay. |
Public hideTubeTracksOnZoom | ||||||||||||
hideTubeTracksOnZoom(scene: Scene, minRadius: number)
|
||||||||||||
Hide tube geometry of tracks on zoom if the camera is too close. (For visibility of vertices)
Parameters :
Returns :
void
|
Public lookAtObject | ||||||||||||||||||||
lookAtObject(uuid: string, objectsGroup: Object3D, offset: number)
|
||||||||||||||||||||
Move the camera to look at the object with the given uuid. with the given uuid.
Parameters :
Returns :
void
|
Private objectIndex | ||||||||||||
objectIndex(obj: OrbitControls, list: OrbitControls[])
|
||||||||||||
Get the index of orbit controls from a list of orbit controls.
Parameters :
Returns :
number
Index of the orbit controls in the given list. Returns -1 if not found. |
Private positionSync | ||||||||
positionSync(controls: OrbitControls)
|
||||||||
Synchronously update position of the orbit controls.
Parameters :
Returns :
void
|
Public removeControls | ||||||||
removeControls(controls: OrbitControls)
|
||||||||
Remove orbit controls from the controls list.
Parameters :
Returns :
void
|
Private rotationSync | ||||||||
rotationSync(controls: OrbitControls)
|
||||||||
Synchronously update rotation of the orbit controls.
Parameters :
Returns :
void
|
setActiveControls | ||||||||
setActiveControls(controls: OrbitControls)
|
||||||||
Set the currently active orbit controls.
Parameters :
Returns :
void
|
setMainControls | ||||||||
setMainControls(controls: OrbitControls)
|
||||||||
Set the main orbit controls.
Parameters :
Returns :
void
|
setOverlayControls | ||||||||
setOverlayControls(controls: OrbitControls)
|
||||||||
Set orbit controls for overlay.
Parameters :
Returns :
void
|
Private setupResize | ||||||||
setupResize(rendererElement: HTMLCanvasElement)
|
||||||||
Set up to make camera(s) adapt to window resize.
Parameters :
Returns :
void
|
Public swapControls |
swapControls()
|
Swap the main and overlay orbit controls.
Returns :
void
|
Public transformSync |
transformSync()
|
Synchronously transform the controls by updating the position and rotation.
Returns :
void
|
Public update | ||||||||
update(controls: OrbitControls)
|
||||||||
Update orbit controls.
Parameters :
Returns :
void
|
Public updateSync |
updateSync()
|
Synchronously update all controls.
Returns :
void
|
Public zoomTo | ||||||||||||
zoomTo(zoomFactor: number, zoomTime: number)
|
||||||||||||
Zoom all the cameras by a specific zoom factor. The factor may either be greater or smaller.
Parameters :
Returns :
void
|
import { Tween } from '@tweenjs/tween.js';
import {
Camera,
PerspectiveCamera,
OrthographicCamera,
Object3D,
Vector3,
Group,
Scene,
Mesh,
TubeGeometry,
} from 'three';
import { OrbitControls } from 'three/examples/jsm/controls/OrbitControls.js';
import { RendererManager } from './renderer-manager';
/**
* Manager for managing event display controls.
*/
export class ControlsManager {
/** Currently active orbit controls. */
private activeControls: OrbitControls;
/** The main orbit controls. */
private mainControls: OrbitControls;
/** Orbit controls for overlay view. */
private overlayControls: OrbitControls;
/** All orbit controls. */
private controls: OrbitControls[];
/** Orbit controls for the perspective view. */
private perspectiveControls: OrbitControls;
/** Orbit controls for the orthographic view. */
private orthographicControls: OrbitControls;
/**
* Constructor for setting up all the controls.
* @param rendererManager The renderer manager to get the main renderer.
*/
constructor(
rendererManager: RendererManager,
defaultView: number[] = [0, 0, 200],
) {
this.controls = [];
// this.mainControls = new OrbitControls();
// this.overlayControls = null;
const rendererElement = rendererManager.getMainRenderer()?.domElement;
// Arguments: FOV, aspect ratio, near and far distances
const perspectiveCamera = new PerspectiveCamera(
75,
rendererElement.offsetWidth / rendererElement.offsetHeight,
10,
100000,
);
// Arguments: left, right, top, bottom, near and far distances
const orthographicCamera = new OrthographicCamera(
rendererElement.offsetWidth / -2,
rendererElement.offsetWidth / 2,
rendererElement.offsetHeight / 2,
rendererElement.offsetHeight / -2,
10,
100000,
);
// Orbit controls allow to move around
this.perspectiveControls = this.setOrbitControls(
perspectiveCamera,
rendererElement,
);
this.orthographicControls = this.setOrbitControls(
orthographicCamera,
rendererElement,
);
perspectiveCamera.position.z = orthographicCamera.position.z =
defaultView[2];
perspectiveCamera.position.y = orthographicCamera.position.y =
defaultView[1];
perspectiveCamera.position.x = orthographicCamera.position.x =
defaultView[0];
if (defaultView.length >= 6) {
this.perspectiveControls.target = this.orthographicControls.target =
new Vector3(defaultView[3], defaultView[4], defaultView[5]);
}
// Set active orbit controls
this.addControls(this.perspectiveControls);
this.addControls(this.orthographicControls);
this.setActiveControls(this.perspectiveControls);
this.setMainControls(this.perspectiveControls);
this.setOverlayControls(this.orthographicControls);
// Add listener
this.getActiveControls().addEventListener('change', () => {
this.transformSync();
this.updateSync();
});
// Modify camera(s) on window resize
this.setupResize(rendererElement);
}
/**
* Set orbit controls for the camera.
* @param camera The camera with which to create the orbit controls.
* @param domElement DOM element of the renderer to associate the orbit controls with.
* @returns Configured orbit controls.
*/
private setOrbitControls(
camera: PerspectiveCamera | OrthographicCamera,
domElement: HTMLElement,
): OrbitControls {
const controls: OrbitControls = new OrbitControls(camera, domElement);
controls.enableDamping = true;
controls.dampingFactor = 0.25;
controls.enableZoom = true;
controls.autoRotate = false;
return controls;
}
// SET/GET
/**
* Set the currently active orbit controls.
* @param controls Orbit controls to be set as active.
*/
setActiveControls(controls: OrbitControls) {
this.activeControls = controls;
}
/**
* Set the main orbit controls.
* @param controls Orbit controls to be set as main.
*/
setMainControls(controls: OrbitControls) {
this.mainControls = controls;
}
/**
* Set orbit controls for overlay.
* @param controls Orbit controls to be set for overlay.
*/
setOverlayControls(controls: OrbitControls) {
this.overlayControls = controls;
}
/**
* Get currently active orbit controls.
* @returns Currently active orbit controls.
*/
getActiveControls(): OrbitControls {
return this.activeControls;
}
/**
* Get the main orbit controls.
* @returns Main orbit controls.
*/
getMainControls(): OrbitControls {
return this.mainControls;
}
/**
* Get orbit controls for overlay.
* @returns Orbit controls for overlay.
*/
getOverlayControls(): OrbitControls {
return this.overlayControls;
}
/**
* Get the currently active camera.
* @returns Currently active camera.
*/
getActiveCamera(): Camera {
return this.activeControls.object as Camera;
}
/**
* Get the main camera.
* @returns Main camera.
*/
getMainCamera(): Camera {
return this.mainControls.object as Camera;
}
/**
* Get the camera for overlay.
* @returns The camera for overlay.
*/
getOverlayCamera(): Camera {
return this.overlayControls.object as Camera;
}
/**
* Get the main and overlay cameras.
* @returns An array containing the main and overlay cameras.
*/
getAllCameras(): Camera[] {
return [this.getMainCamera(), this.getOverlayCamera()];
}
// FUNCTIONS
/**
* Add orbit controls to the controls list.
* @param controls Orbit controls to be added.
*/
public addControls(controls: OrbitControls): void {
if (!this.containsObject(controls, this.controls)) {
this.controls.push(controls);
}
}
/**
* Remove orbit controls from the controls list.
* @param controls Orbit controls to be removed.
*/
public removeControls(controls: OrbitControls): void {
const index: number = this.controls.indexOf(controls);
if (index > -1) {
this.controls.splice(index, 1);
}
}
/**
* Swap the main and overlay orbit controls.
*/
public swapControls(): void {
const temp: OrbitControls = this.mainControls;
this.mainControls = this.overlayControls;
// this._mainControls.autoRotate = temp.autoRotate;
this.overlayControls = temp;
}
/**
* Synchronously update all controls.
*/
public updateSync(): void {
for (const control of this.controls) {
if (control === this.activeControls) {
continue;
}
this.update(control);
}
}
/**
* Update orbit controls.
* @param controls Orbit controls to be updated.
*/
public update(controls: OrbitControls): void {
controls.update();
}
/**
* Synchronously transform the controls by updating the position and rotation.
*/
public transformSync(): void {
for (const control of this.controls) {
if (control === this.activeControls) {
continue;
}
this.positionSync(control);
this.rotationSync(control);
}
}
/**
* Zoom all the cameras by a specific zoom factor.
* The factor may either be greater or smaller.
* @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) {
for (const camera of this.getAllCameras()) {
if (camera instanceof OrthographicCamera) {
const animation = new Tween(camera);
animation
.to(
{
zoom: camera.zoom * (1 / zoomFactor),
},
zoomTime,
)
.start();
camera.updateProjectionMatrix();
} else {
const animation = new Tween(camera.position);
animation
.to(
{
x: camera.position.x * zoomFactor,
y: camera.position.y * zoomFactor,
z: camera.position.z * zoomFactor,
},
zoomTime,
)
.start();
}
}
}
/**
* Move the camera to look at the object with the given uuid.
* @param uuid uuid of the object.
* @param objectsGroup Group of objects to be traversed for finding the object
* with the given uuid.
*/
public lookAtObject(
uuid: string,
objectsGroup: Object3D,
offset: number = 0,
) {
const origin = new Vector3(0, 0, 0);
const objectPosition = this.getObjectPosition(uuid, objectsGroup);
if (objectPosition) {
// Check if the object is away from the origin
for (const camera of this.getAllCameras()) {
// Moving the camera to the object's position and then zooming out
new Tween(camera.position)
.to(
{
x: objectPosition.x * 1.1 + offset,
y: objectPosition.y * 1.1 + offset,
z: objectPosition.z * 1.1 + offset,
},
200,
)
.start();
}
}
}
/**
* Get position of object from UUID.
* @param uuid UUID of the object.
* @param objectsGroup Objects group to look into for the object.
* @returns Position of the 3D object.
*/
public getObjectPosition(uuid: string, objectsGroup: Object3D): Vector3 {
const object = objectsGroup.getObjectByProperty('uuid', uuid) as any;
if (object) {
const origin = new Vector3(0, 0, 0);
let objectPosition = new Vector3();
if (object instanceof Group) {
// If it is a group of other event data we traverse through it
object.traverse((childObject: any) => {
// Make sure the child is not a group (e.g Track is a group)
if (childObject.children.length === 0) {
if (childObject.position.equals(origin)) {
// Get the max vector from the bounding box to accumulate with the clusters
if (childObject.geometry?.boundingSphere) {
try {
objectPosition.add(
childObject.geometry.boundingSphere.getBoundingBox().max,
);
} catch (e) {
objectPosition.add(
childObject.geometry.boundingSphere?.center,
);
}
}
} else {
objectPosition.add(childObject.position);
}
}
});
} else if (object.position.equals(origin)) {
// Get the center of bounding sphere of objects with no position
objectPosition = object.geometry?.boundingSphere?.center;
} else {
// Get the object position for all other elements
objectPosition = object.position;
}
return objectPosition;
} else {
return new Vector3();
}
}
/**
* Hide tube geometry of tracks on zoom if the camera is too close.
* (For visibility of vertices)
* @param scene Scene to look in for tracks.
* @param minRadius Radius after which the tube tracks should be invisible.
*/
public hideTubeTracksOnZoom(scene: Scene, minRadius: number) {
let tracksHidden = false;
const origin = new Vector3();
this.activeControls.addEventListener('change', (event) => {
const isCameraClose =
(event?.target?.object?.position as Vector3).distanceTo(origin) <
minRadius;
if (isCameraClose && !tracksHidden) {
scene.getObjectByName('Tracks')?.traverse((track) => {
if (
track.name === 'Track' &&
(track as Mesh).geometry instanceof TubeGeometry
) {
track.visible = false;
}
});
tracksHidden = true;
} else if (!isCameraClose && tracksHidden) {
scene.getObjectByName('Tracks')?.traverse((track) => {
if (
track.name === 'Track' &&
(track as Mesh).geometry instanceof TubeGeometry
) {
track.visible = true;
}
});
tracksHidden = false;
}
});
}
/**
* Synchronously update position of the orbit controls.
* @param controls Orbit controls whose position is to be updated.
*/
private positionSync(controls: OrbitControls): void {
controls.object.position.set(
this.activeControls.object.position.x,
this.activeControls.object.position.y,
this.activeControls.object.position.z,
);
// controls.update();
}
/**
* Synchronously update rotation of the orbit controls.
* @param controls Controls whose rotation is to be updated.
*/
private rotationSync(controls: OrbitControls): void {
controls.target = this.activeControls.target;
// controls.update();
}
/**
* Check if the list of orbit controls contains a specific orbit controls.
* @param obj Orbit controls to be checked for containment.
* @param list List of orbit controls.
* @returns If the list contains the orbit controls.
*/
private containsObject(obj: OrbitControls, list: OrbitControls[]): boolean {
for (const object of list) {
if (object === obj) {
return true;
}
}
return false;
}
/**
* Set up to make camera(s) adapt to window resize.
* @param rendererElement Canvas element of the main renderer.
*/
private setupResize(rendererElement: HTMLCanvasElement) {
window.addEventListener('resize', () => {
let mainCamera = this.getMainCamera() as any;
if (mainCamera.isOrthographicCamera) {
mainCamera = mainCamera as OrthographicCamera;
mainCamera.left = rendererElement.offsetWidth / -2;
mainCamera.right = rendererElement.offsetWidth / 2;
mainCamera.top = rendererElement.offsetHeight / 2;
mainCamera.bottom = rendererElement.offsetHeight / -2;
mainCamera.updateProjectionMatrix();
} else {
mainCamera = mainCamera as PerspectiveCamera;
mainCamera.aspect =
rendererElement.offsetWidth / rendererElement.offsetHeight;
mainCamera.updateProjectionMatrix();
}
});
}
/**
* Get the index of orbit controls from a list of orbit controls.
* @param obj Orbit controls whose index is to be obtained.
* @param list List of orbit controls.
* @returns Index of the orbit controls in the given list. Returns -1 if not found.
*/
private objectIndex(obj: OrbitControls, list: OrbitControls[]): number {
for (let i = 0; i < list.length; i++) {
if (list[i] === obj) {
return i;
}
}
return -1;
}
}