2019-02-11 13:43:04 +01:00
|
|
|
import { Position, Size, UnknownObject } from "./types";
|
|
|
|
import {
|
|
|
|
sizePropsDecoder,
|
|
|
|
positionPropsDecoder,
|
|
|
|
parseIntOr,
|
2019-03-08 12:51:03 +01:00
|
|
|
parseBoolean,
|
|
|
|
notEmptyStringOr
|
2019-02-11 13:43:04 +01:00
|
|
|
} from "./lib";
|
|
|
|
import TypedEvent, { Listener, Disposable } from "./TypedEvent";
|
|
|
|
|
2019-02-18 17:23:57 +01:00
|
|
|
// Enum: https://www.typescriptlang.org/docs/handbook/enums.html.
|
2019-03-08 12:51:03 +01:00
|
|
|
export const enum ItemType {
|
2019-02-18 17:23:57 +01:00
|
|
|
STATIC_GRAPH = 0,
|
|
|
|
MODULE_GRAPH = 1,
|
|
|
|
SIMPLE_VALUE = 2,
|
|
|
|
PERCENTILE_BAR = 3,
|
|
|
|
LABEL = 4,
|
|
|
|
ICON = 5,
|
|
|
|
SIMPLE_VALUE_MAX = 6,
|
|
|
|
SIMPLE_VALUE_MIN = 7,
|
|
|
|
SIMPLE_VALUE_AVG = 8,
|
|
|
|
PERCENTILE_BUBBLE = 9,
|
|
|
|
SERVICE = 10,
|
|
|
|
GROUP_ITEM = 11,
|
|
|
|
BOX_ITEM = 12,
|
|
|
|
LINE_ITEM = 13,
|
|
|
|
AUTO_SLA_GRAPH = 14,
|
|
|
|
CIRCULAR_PROGRESS_BAR = 15,
|
|
|
|
CIRCULAR_INTERIOR_PROGRESS_BAR = 16,
|
|
|
|
DONUT_GRAPH = 17,
|
|
|
|
BARS_GRAPH = 18,
|
|
|
|
CLOCK = 19,
|
|
|
|
COLOR_CLOUD = 20
|
|
|
|
}
|
|
|
|
|
2019-02-11 13:43:04 +01:00
|
|
|
// Base item properties. This interface should be extended by the item implementations.
|
2019-03-08 12:51:03 +01:00
|
|
|
export interface ItemProps extends Position, Size {
|
2019-02-11 13:43:04 +01:00
|
|
|
readonly id: number;
|
2019-03-08 12:51:03 +01:00
|
|
|
readonly type: ItemType;
|
2019-02-11 13:43:04 +01:00
|
|
|
label: string | null;
|
2019-02-18 17:23:57 +01:00
|
|
|
labelPosition: "up" | "right" | "down" | "left";
|
2019-02-11 13:43:04 +01:00
|
|
|
isLinkEnabled: boolean;
|
|
|
|
isOnTop: boolean;
|
|
|
|
parentId: number | null;
|
|
|
|
aclGroupId: number | null;
|
|
|
|
}
|
|
|
|
|
2019-02-18 17:23:57 +01:00
|
|
|
// FIXME: Fix type compatibility.
|
2019-03-08 12:51:03 +01:00
|
|
|
export interface ItemClickEvent<Props extends ItemProps> {
|
|
|
|
// data: Props;
|
2019-02-18 17:23:57 +01:00
|
|
|
data: UnknownObject;
|
2019-02-26 17:05:30 +01:00
|
|
|
}
|
2019-02-18 17:23:57 +01:00
|
|
|
|
|
|
|
/**
|
2019-03-07 18:41:33 +01:00
|
|
|
* Extract a valid enum value from a raw label positi9on value.
|
2019-02-18 17:23:57 +01:00
|
|
|
* @param labelPosition Raw value.
|
|
|
|
*/
|
2019-02-26 17:05:30 +01:00
|
|
|
const parseLabelPosition = (
|
|
|
|
labelPosition: any // eslint-disable-line @typescript-eslint/no-explicit-any
|
2019-03-08 12:51:03 +01:00
|
|
|
): ItemProps["labelPosition"] => {
|
2019-02-18 17:23:57 +01:00
|
|
|
switch (labelPosition) {
|
|
|
|
case "up":
|
|
|
|
case "right":
|
|
|
|
case "down":
|
|
|
|
case "left":
|
|
|
|
return labelPosition;
|
|
|
|
default:
|
|
|
|
return "down";
|
|
|
|
}
|
2019-02-11 13:43:04 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Build a valid typed object from a raw object.
|
|
|
|
* This will allow us to ensure the type safety.
|
|
|
|
*
|
|
|
|
* @param data Raw object.
|
2019-02-11 17:35:16 +01:00
|
|
|
* @return An object representing the item props.
|
2019-02-11 13:43:04 +01:00
|
|
|
* @throws Will throw a TypeError if some property
|
|
|
|
* is missing from the raw object or have an invalid type.
|
|
|
|
*/
|
2019-03-08 12:51:03 +01:00
|
|
|
export function itemBasePropsDecoder(data: UnknownObject): ItemProps | never {
|
2019-02-11 13:43:04 +01:00
|
|
|
if (data.id == null || isNaN(parseInt(data.id))) {
|
|
|
|
throw new TypeError("invalid id.");
|
|
|
|
}
|
|
|
|
if (data.type == null || isNaN(parseInt(data.type))) {
|
|
|
|
throw new TypeError("invalid type.");
|
|
|
|
}
|
|
|
|
|
|
|
|
return {
|
|
|
|
id: parseInt(data.id),
|
|
|
|
type: parseInt(data.type),
|
2019-03-08 12:51:03 +01:00
|
|
|
label: notEmptyStringOr(data.label, null),
|
2019-02-18 17:23:57 +01:00
|
|
|
labelPosition: parseLabelPosition(data.labelPosition),
|
2019-02-11 13:43:04 +01:00
|
|
|
isLinkEnabled: parseBoolean(data.isLinkEnabled),
|
|
|
|
isOnTop: parseBoolean(data.isOnTop),
|
|
|
|
parentId: parseIntOr(data.parentId, null),
|
|
|
|
aclGroupId: parseIntOr(data.aclGroupId, null),
|
|
|
|
...sizePropsDecoder(data), // Object spread. It will merge the properties of the two objects.
|
|
|
|
...positionPropsDecoder(data) // Object spread. It will merge the properties of the two objects.
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2019-03-28 17:59:35 +01:00
|
|
|
/**
|
|
|
|
* Base class of the visual console items. Should be extended to use its capabilities.
|
|
|
|
*/
|
2019-03-08 12:51:03 +01:00
|
|
|
abstract class VisualConsoleItem<Props extends ItemProps> {
|
2019-02-11 13:43:04 +01:00
|
|
|
// Properties of the item.
|
2019-03-08 12:51:03 +01:00
|
|
|
private itemProps: Props;
|
2019-02-18 17:23:57 +01:00
|
|
|
// Reference to the DOM element which will contain the item.
|
|
|
|
public readonly elementRef: HTMLElement;
|
2019-03-28 17:59:35 +01:00
|
|
|
private readonly labelElementRef: HTMLElement;
|
2019-02-18 17:23:57 +01:00
|
|
|
// Reference to the DOM element which will contain the view of the item which extends this class.
|
|
|
|
protected readonly childElementRef: HTMLElement;
|
2019-02-11 13:43:04 +01:00
|
|
|
// Event manager for click events.
|
2019-03-08 12:51:03 +01:00
|
|
|
private readonly clickEventManager = new TypedEvent<ItemClickEvent<Props>>();
|
2019-02-11 13:43:04 +01:00
|
|
|
// List of references to clean the event listeners.
|
|
|
|
private readonly disposables: Disposable[] = [];
|
|
|
|
|
|
|
|
/**
|
|
|
|
* To create a new element which will be inside the item box.
|
|
|
|
* @return Item.
|
|
|
|
*/
|
|
|
|
abstract createDomElement(): HTMLElement;
|
|
|
|
|
2019-03-08 12:51:03 +01:00
|
|
|
public constructor(props: Props) {
|
2019-01-30 12:06:35 +01:00
|
|
|
this.itemProps = props;
|
2019-02-11 13:43:04 +01:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Get a HTMLElement which represents the container box
|
|
|
|
* of the Visual Console item. This element will manage
|
|
|
|
* all the common things like click events, show a border
|
|
|
|
* when hovered, etc.
|
|
|
|
*/
|
2019-02-18 17:23:57 +01:00
|
|
|
this.elementRef = this.createContainerDomElement();
|
2019-04-04 18:47:02 +02:00
|
|
|
this.labelElementRef = this.createLabelDomElement();
|
2019-02-11 13:43:04 +01:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Get a HTMLElement which represents the custom view
|
|
|
|
* of the Visual Console item. This element will be
|
|
|
|
* different depending on the item implementation.
|
|
|
|
*/
|
2019-02-18 17:23:57 +01:00
|
|
|
this.childElementRef = this.createDomElement();
|
2019-02-11 13:43:04 +01:00
|
|
|
|
2019-02-18 17:23:57 +01:00
|
|
|
// Insert the elements into the container.
|
2019-03-29 11:54:01 +01:00
|
|
|
this.elementRef.append(this.childElementRef, this.labelElementRef);
|
|
|
|
|
|
|
|
// Resize element.
|
|
|
|
this.resizeElement(props.width, props.height);
|
|
|
|
// Set label position.
|
|
|
|
this.changeLabelPosition(props.labelPosition);
|
2019-02-11 13:43:04 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* To create a new box for the visual console item.
|
|
|
|
* @return Item box.
|
|
|
|
*/
|
2019-02-18 17:23:57 +01:00
|
|
|
private createContainerDomElement(): HTMLElement {
|
2019-02-11 13:43:04 +01:00
|
|
|
const box: HTMLDivElement = document.createElement("div");
|
|
|
|
box.className = "visual-console-item";
|
2019-03-29 11:54:01 +01:00
|
|
|
// box.style.width = `${this.props.width}px`;
|
|
|
|
// box.style.height = `${this.props.height}px`;
|
2019-02-18 17:23:57 +01:00
|
|
|
box.style.left = `${this.props.x}px`;
|
|
|
|
box.style.top = `${this.props.y}px`;
|
2019-02-11 13:43:04 +01:00
|
|
|
box.onclick = () => this.clickEventManager.emit({ data: this.props });
|
2019-03-28 17:59:35 +01:00
|
|
|
|
2019-02-11 13:43:04 +01:00
|
|
|
return box;
|
2019-01-30 12:06:35 +01:00
|
|
|
}
|
|
|
|
|
2019-04-04 18:47:02 +02:00
|
|
|
/**
|
|
|
|
* To create a new label for the visual console item.
|
|
|
|
* @return Item label.
|
|
|
|
*/
|
|
|
|
protected createLabelDomElement(): HTMLElement {
|
|
|
|
const element = document.createElement("div");
|
|
|
|
element.className = "visual-console-item-label";
|
|
|
|
// Add the label if it exists.
|
|
|
|
if (this.props.label && this.props.label.length) {
|
|
|
|
element.innerHTML = this.props.label;
|
|
|
|
}
|
|
|
|
|
|
|
|
return element;
|
|
|
|
}
|
|
|
|
|
2019-02-11 13:43:04 +01:00
|
|
|
/**
|
|
|
|
* Public accessor of the `props` property.
|
|
|
|
* @return Properties.
|
|
|
|
*/
|
2019-03-08 12:51:03 +01:00
|
|
|
public get props(): Props {
|
2019-01-30 12:06:35 +01:00
|
|
|
return this.itemProps;
|
|
|
|
}
|
2019-02-11 13:43:04 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Public setter of the `props` property.
|
|
|
|
* If the new props are different enough than the
|
|
|
|
* stored props, a render would be fired.
|
|
|
|
* @param newProps
|
|
|
|
*/
|
2019-03-08 12:51:03 +01:00
|
|
|
public set props(newProps: Props) {
|
2019-02-11 13:43:04 +01:00
|
|
|
const prevProps = this.props;
|
|
|
|
// Update the internal props.
|
|
|
|
this.itemProps = newProps;
|
|
|
|
|
|
|
|
// From this point, things which rely on this.props can access to the changes.
|
|
|
|
|
|
|
|
// Check if we should re-render.
|
2019-03-19 10:09:18 +01:00
|
|
|
if (this.shouldBeUpdated(prevProps, newProps)) this.render(prevProps);
|
2019-02-11 13:43:04 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* To compare the previous and the new props and returns a boolean value
|
|
|
|
* in case the difference is meaningfull enough to perform DOM changes.
|
|
|
|
*
|
|
|
|
* Here, the only comparision is done by reference.
|
|
|
|
*
|
|
|
|
* Override this function to perform a different comparision depending on the item needs.
|
|
|
|
*
|
2019-03-19 10:09:18 +01:00
|
|
|
* @param prevProps
|
2019-02-11 13:43:04 +01:00
|
|
|
* @param newProps
|
|
|
|
* @return Whether the difference is meaningful enough to perform DOM changes or not.
|
|
|
|
*/
|
2019-03-19 10:09:18 +01:00
|
|
|
protected shouldBeUpdated(prevProps: Props, newProps: Props): boolean {
|
|
|
|
return prevProps !== newProps;
|
2019-02-11 13:43:04 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* To recreate or update the HTMLElement which represents the item into the DOM.
|
2019-02-18 17:23:57 +01:00
|
|
|
* @param prevProps If exists it will be used to only perform DOM updates instead of a full replace.
|
2019-02-11 13:43:04 +01:00
|
|
|
*/
|
2019-03-08 12:51:03 +01:00
|
|
|
public render(prevProps: Props | null = null): void {
|
2019-03-29 11:54:01 +01:00
|
|
|
this.childElementRef.innerHTML = this.createDomElement().innerHTML;
|
|
|
|
|
2019-02-11 13:43:04 +01:00
|
|
|
// Move box.
|
2019-03-19 10:09:18 +01:00
|
|
|
if (!prevProps || this.positionChanged(prevProps, this.props)) {
|
|
|
|
this.moveElement(this.props.x, this.props.y);
|
2019-02-11 13:43:04 +01:00
|
|
|
}
|
|
|
|
// Resize box.
|
2019-03-19 10:09:18 +01:00
|
|
|
if (!prevProps || this.sizeChanged(prevProps, this.props)) {
|
|
|
|
this.resizeElement(this.props.width, this.props.height);
|
2019-02-11 13:43:04 +01:00
|
|
|
}
|
2019-03-28 17:59:35 +01:00
|
|
|
// Change label.
|
2019-03-29 11:54:01 +01:00
|
|
|
if (!prevProps || prevProps.label !== this.props.label) {
|
2019-04-04 18:47:02 +02:00
|
|
|
this.labelElementRef.innerHTML = this.createLabelDomElement().innerHTML;
|
2019-03-29 11:54:01 +01:00
|
|
|
}
|
|
|
|
// Change label position.
|
|
|
|
if (!prevProps || prevProps.labelPosition !== this.props.labelPosition) {
|
|
|
|
this.changeLabelPosition(this.props.labelPosition);
|
2019-03-28 17:59:35 +01:00
|
|
|
}
|
2019-02-11 13:43:04 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* To remove the event listeners and the elements from the DOM.
|
|
|
|
*/
|
2019-02-26 17:05:30 +01:00
|
|
|
public remove(): void {
|
2019-02-11 13:43:04 +01:00
|
|
|
// Event listeners.
|
|
|
|
this.disposables.forEach(_ => _.dispose());
|
|
|
|
// VisualConsoleItem DOM element.
|
2019-02-18 17:23:57 +01:00
|
|
|
this.elementRef.remove();
|
2019-02-11 13:43:04 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2019-03-19 10:09:18 +01:00
|
|
|
* Compare the previous and the new position and return
|
|
|
|
* a boolean value in case the position changed.
|
|
|
|
* @param prevPosition
|
|
|
|
* @param newPosition
|
|
|
|
* @return Whether the position changed or not.
|
|
|
|
*/
|
|
|
|
protected positionChanged(
|
|
|
|
prevPosition: Position,
|
|
|
|
newPosition: Position
|
|
|
|
): boolean {
|
|
|
|
return prevPosition.x !== newPosition.x || prevPosition.y !== newPosition.y;
|
|
|
|
}
|
|
|
|
|
2019-03-29 11:54:01 +01:00
|
|
|
/**
|
|
|
|
* Move the label around the item content.
|
|
|
|
* @param position Label position.
|
|
|
|
*/
|
|
|
|
protected changeLabelPosition(position: Props["labelPosition"]): void {
|
|
|
|
switch (position) {
|
|
|
|
case "up":
|
|
|
|
this.elementRef.style.flexDirection = "column-reverse";
|
|
|
|
break;
|
|
|
|
case "left":
|
|
|
|
this.elementRef.style.flexDirection = "row-reverse";
|
|
|
|
break;
|
|
|
|
case "right":
|
|
|
|
this.elementRef.style.flexDirection = "row";
|
|
|
|
break;
|
|
|
|
case "down":
|
|
|
|
default:
|
|
|
|
this.elementRef.style.flexDirection = "column";
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-03-19 10:09:18 +01:00
|
|
|
/**
|
|
|
|
* Move the DOM container.
|
2019-02-11 13:43:04 +01:00
|
|
|
* @param x Horizontal axis position.
|
|
|
|
* @param y Vertical axis position.
|
|
|
|
*/
|
2019-03-19 10:09:18 +01:00
|
|
|
protected moveElement(x: number, y: number): void {
|
2019-02-18 17:23:57 +01:00
|
|
|
this.elementRef.style.left = `${x}px`;
|
|
|
|
this.elementRef.style.top = `${y}px`;
|
2019-02-11 13:43:04 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2019-03-19 10:09:18 +01:00
|
|
|
* Update the position into the properties and move the DOM container.
|
|
|
|
* @param x Horizontal axis position.
|
|
|
|
* @param y Vertical axis position.
|
2019-02-11 13:43:04 +01:00
|
|
|
*/
|
2019-03-19 10:09:18 +01:00
|
|
|
public move(x: number, y: number): void {
|
|
|
|
this.moveElement(x, y);
|
|
|
|
this.itemProps = {
|
|
|
|
...this.props, // Object spread: http://es6-features.org/#SpreadOperator
|
|
|
|
x,
|
|
|
|
y
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Compare the previous and the new size and return
|
|
|
|
* a boolean value in case the size changed.
|
|
|
|
* @param prevSize
|
|
|
|
* @param newSize
|
|
|
|
* @return Whether the size changed or not.
|
|
|
|
*/
|
|
|
|
protected sizeChanged(prevSize: Size, newSize: Size): boolean {
|
|
|
|
return (
|
|
|
|
prevSize.width !== newSize.width || prevSize.height !== newSize.height
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2019-03-28 17:59:35 +01:00
|
|
|
* Resize the DOM content container.
|
2019-03-19 10:09:18 +01:00
|
|
|
* @param width
|
|
|
|
* @param height
|
|
|
|
*/
|
|
|
|
protected resizeElement(width: number, height: number): void {
|
2019-03-29 11:54:01 +01:00
|
|
|
// The most valuable size is the content size.
|
2019-04-08 12:54:29 +02:00
|
|
|
this.childElementRef.style.width = width > 0 ? `${width}px` : null;
|
|
|
|
this.childElementRef.style.height = height > 0 ? `${height}px` : null;
|
2019-02-11 13:43:04 +01:00
|
|
|
}
|
|
|
|
|
2019-03-19 10:09:18 +01:00
|
|
|
/**
|
|
|
|
* Update the size into the properties and resize the DOM container.
|
|
|
|
* @param width
|
|
|
|
* @param height
|
|
|
|
*/
|
|
|
|
public resize(width: number, height: number): void {
|
|
|
|
this.resizeElement(width, height);
|
|
|
|
this.itemProps = {
|
|
|
|
...this.props, // Object spread: http://es6-features.org/#SpreadOperator
|
|
|
|
width,
|
|
|
|
height
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2019-02-11 13:43:04 +01:00
|
|
|
/**
|
|
|
|
* To add an event handler to the click of the linked visual console elements.
|
|
|
|
* @param listener Function which is going to be executed when a linked console is clicked.
|
|
|
|
*/
|
2019-03-08 12:51:03 +01:00
|
|
|
public onClick(listener: Listener<ItemClickEvent<Props>>): void {
|
2019-02-11 13:43:04 +01:00
|
|
|
/*
|
|
|
|
* The '.on' function returns a function which will clean the event
|
|
|
|
* listener when executed. We store all the 'dispose' functions to
|
|
|
|
* call them when the item should be cleared.
|
|
|
|
*/
|
|
|
|
this.disposables.push(this.clickEventManager.on(listener));
|
|
|
|
}
|
2019-01-30 12:06:35 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
export default VisualConsoleItem;
|