immersive2/src/integration/indexdbPersistenceManager.ts

190 lines
7.1 KiB
TypeScript

import {DiagramListing, DiagramListingEvent, DiagramListingEventType, IPersistenceManager} from "./iPersistenceManager";
import {AbstractMesh, Observable, Vector3} from "@babylonjs/core";
import {DiagramEntity} from "../diagram/diagramEntity";
import Dexie from "dexie";
import {MeshConverter} from "../diagram/meshConverter";
import log from "loglevel";
import {AppConfigType} from "../util/appConfigType";
export class IndexdbPersistenceManager implements IPersistenceManager {
private readonly logger = log.getLogger('IndexdbPersistenceManager');
public readonly diagramListingObserver: Observable<DiagramListingEvent> = new Observable<DiagramListingEvent>();
public readonly updateObserver: Observable<DiagramEntity> = new Observable<DiagramEntity>();
public readonly configObserver: Observable<AppConfigType> = new Observable<AppConfigType>();
private db: Dexie;
private currentDiagramId: string;
constructor(name: string) {
this.db = new Dexie(name);
const version = 6;
this.db.version(version).stores({config: "id,gridSnap,rotateSnap,createSnap"});
this.db.version(version).stores({entities: "id,diagramlistingid,position,rotation,last_seen,template,text,scale,color"});
this.db.version(version).stores({diagramlisting: "id,name,description,sharekey"});
this.db.version(version).stores({newRelicData: "id,priority, incidentId"});
this.logger.debug("IndexdbPersistenceManager constructed");
}
public setCurrentDiagram(diagram: DiagramListing) {
this.currentDiagramId = diagram.id;
}
public add(mesh: AbstractMesh) {
if (!mesh) {
this.logger.error("Adding null mesh, early return");
return;
}
const entity = <any>MeshConverter.toDiagramEntity(mesh);
entity.position = this.vectoxys(mesh.position);
entity.rotation = this.vectoxys(mesh.rotation);
entity.scale = this.vectoxys(mesh.scaling);
entity.diagramlistingid = this.currentDiagramId;
this.db["entities"].add(entity);
this.logger.debug('add', mesh, entity);
}
public addDiagram(diagram: DiagramListing) {
this.db["diagramlisting"].add(diagram);
const event = {
type: DiagramListingEventType.ADD,
listing: diagram
}
this.diagramListingObserver.notifyObservers(event);
}
public remove(mesh: AbstractMesh) {
if (!mesh) {
this.logger.error("Removing null mesh, early return");
return;
}
this.db["entities"].delete(mesh.id);
}
public setConfig(config: AppConfigType) {
config.id = 1;
this.db["config"].put(config);
this.logger.debug('setConfig', config);
this.configObserver.notifyObservers(config);
}
public removeDiagram(diagram: DiagramListing) {
this.db["diagramlisting"].delete(diagram.id);
const event = {
type: DiagramListingEventType.REMOVE,
listing: diagram
}
this.diagramListingObserver.notifyObservers(event);
}
modifyDiagram(diagram: DiagramListing) {
this.db["diagramlisting"].update(diagram.id, diagram);
const event = {
type: DiagramListingEventType.MODIFY,
listing: diagram
}
this.diagramListingObserver.notifyObservers(event);
}
public async setNewRelicData(data: any[]) {
this.db["newRelicData"].clear();
data.forEach((d) => {
this.db["newRelicData"].add(d);
});
}
public async getNewRelicData(): Promise<any[]> {
return this.db["newRelicData"].toArray();
}
public async initialize() {
this.logger.info('initialize', this.db['entities'].length);
const configs = await this.db['config'].toArray();
const config = configs[0];
if (config) {
this.logger.debug('initialize config', config);
this.configObserver.notifyObservers(config);
if (config.currentDiagramId) {
this.logger.debug('initialize currentDiagramId', config.currentDiagramId);
const currentDiagram = await this.db['diagramlisting'].get(config.currentDiagramId);
if (currentDiagram) {
this.logger.debug('found currentDiagram', currentDiagram);
this.currentDiagramId = currentDiagram.id;
} else {
this.logger.error('could not find currentDiagram', config.currentDiagramId);
}
} else {
this.logger.warn('no currentDiagramId, using default');
}
}
this.getFilteredEntities().each((e) => {
e.position = this.xyztovec(e.position);
e.rotation = this.xyztovec(e.rotation);
e.scale = this.xyztovec(e.scale);
this.logger.debug('adding', e);
this.updateObserver.notifyObservers(e);
});
this.listDiagrams();
this.logger.info("initialize finished");
}
public modify(mesh) {
if (!mesh) {
this.logger.error("Modifying null mesh, early return");
return;
}
const entity = <any>MeshConverter.toDiagramEntity(mesh);
if (!entity) {
this.logger.error("Modifying null mesh, early return");
return;
}
entity.position = this.vectoxys(mesh.position);
entity.rotation = this.vectoxys(mesh.rotation);
entity.scale = this.vectoxys(mesh.scaling);
this.db["entities"].update(mesh.id, entity);
this.logger.debug('modify', mesh, entity);
}
public changeColor(oldColor, newColor) {
if (!oldColor) {
if (!newColor) {
this.logger.error("changeColor called with no new color, early return");
} else {
this.logger.info("changeColor called with no old Color, new color added to diagram, early return");
}
return;
}
this.logger.debug(`changeColor ${oldColor.toHexString()} to ${newColor.toHexString()}`);
this.getFilteredEntities().filter((e) => e.color == oldColor.toHexString()).modify({color: newColor.toHexString()});
}
private listDiagrams() {
return this.db["diagramlisting"].toArray((diagrams) => {
this.logger.debug('listDiagrams', diagrams);
for (const diagram of diagrams) {
const event = {
type: DiagramListingEventType.GET,
listing: diagram
}
this.diagramListingObserver.notifyObservers(event);
}
});
}
private getFilteredEntities() {
return this.db['entities'].filter((e) => {
if (!this.currentDiagramId && !e.diagramlistingid) {
return true;
} else {
return e.diagramlistingid == this.currentDiagramId;
}
}
);
}
private vectoxys(v: Vector3): { x, y, z } {
return {x: v.x, y: v.y, z: v.z};
}
private xyztovec(xyz: { x, y, z }): Vector3 {
return new Vector3(xyz.x, xyz.y, xyz.z);
}
}