You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
344 lines
22 KiB
JavaScript
344 lines
22 KiB
JavaScript
/**
|
|
* @fileoverview added by tsickle
|
|
* @suppress {checkTypes,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
|
|
*/
|
|
import * as tslib_1 from "tslib";
|
|
import { InjectionToken } from '@angular/core';
|
|
/** @type {?} */
|
|
export var FC_NODE_COMPONENT_CONFIG = new InjectionToken('fc-node.component.config');
|
|
/**
|
|
* @record
|
|
*/
|
|
export function FcNodeComponentConfig() { }
|
|
if (false) {
|
|
/** @type {?} */
|
|
FcNodeComponentConfig.prototype.nodeComponentType;
|
|
}
|
|
/** @type {?} */
|
|
var htmlPrefix = 'fc';
|
|
/** @type {?} */
|
|
var leftConnectorType = 'leftConnector';
|
|
/** @type {?} */
|
|
var rightConnectorType = 'rightConnector';
|
|
/** @type {?} */
|
|
export var FlowchartConstants = {
|
|
htmlPrefix: htmlPrefix,
|
|
leftConnectorType: leftConnectorType,
|
|
rightConnectorType: rightConnectorType,
|
|
curvedStyle: 'curved',
|
|
lineStyle: 'line',
|
|
dragAnimationRepaint: 'repaint',
|
|
dragAnimationShadow: 'shadow',
|
|
canvasClass: htmlPrefix + '-canvas',
|
|
selectedClass: htmlPrefix + '-selected',
|
|
editClass: htmlPrefix + '-edit',
|
|
activeClass: htmlPrefix + '-active',
|
|
hoverClass: htmlPrefix + '-hover',
|
|
draggingClass: htmlPrefix + '-dragging',
|
|
edgeClass: htmlPrefix + '-edge',
|
|
edgeLabelClass: htmlPrefix + '-edge-label',
|
|
connectorClass: htmlPrefix + '-connector',
|
|
magnetClass: htmlPrefix + '-magnet',
|
|
nodeClass: htmlPrefix + '-node',
|
|
nodeOverlayClass: htmlPrefix + '-node-overlay',
|
|
leftConnectorClass: htmlPrefix + '-' + leftConnectorType + 's',
|
|
rightConnectorClass: htmlPrefix + '-' + rightConnectorType + 's',
|
|
canvasResizeThreshold: 200,
|
|
canvasResizeStep: 200
|
|
};
|
|
/**
|
|
* @record
|
|
*/
|
|
export function FcCoords() { }
|
|
if (false) {
|
|
/** @type {?|undefined} */
|
|
FcCoords.prototype.x;
|
|
/** @type {?|undefined} */
|
|
FcCoords.prototype.y;
|
|
}
|
|
/**
|
|
* @record
|
|
*/
|
|
export function FcRectBox() { }
|
|
if (false) {
|
|
/** @type {?} */
|
|
FcRectBox.prototype.top;
|
|
/** @type {?} */
|
|
FcRectBox.prototype.left;
|
|
/** @type {?} */
|
|
FcRectBox.prototype.right;
|
|
/** @type {?} */
|
|
FcRectBox.prototype.bottom;
|
|
}
|
|
/**
|
|
* @record
|
|
*/
|
|
export function FcConnector() { }
|
|
if (false) {
|
|
/** @type {?} */
|
|
FcConnector.prototype.id;
|
|
/** @type {?} */
|
|
FcConnector.prototype.type;
|
|
}
|
|
/**
|
|
* @record
|
|
*/
|
|
export function FcNode() { }
|
|
if (false) {
|
|
/** @type {?} */
|
|
FcNode.prototype.id;
|
|
/** @type {?} */
|
|
FcNode.prototype.name;
|
|
/** @type {?} */
|
|
FcNode.prototype.connectors;
|
|
/** @type {?|undefined} */
|
|
FcNode.prototype.readonly;
|
|
/* Skipping unhandled member: [key: string]: any;*/
|
|
}
|
|
/**
|
|
* @record
|
|
*/
|
|
export function FcEdge() { }
|
|
if (false) {
|
|
/** @type {?|undefined} */
|
|
FcEdge.prototype.label;
|
|
/** @type {?|undefined} */
|
|
FcEdge.prototype.source;
|
|
/** @type {?|undefined} */
|
|
FcEdge.prototype.destination;
|
|
/** @type {?|undefined} */
|
|
FcEdge.prototype.active;
|
|
}
|
|
/**
|
|
* @record
|
|
*/
|
|
export function FcItemInfo() { }
|
|
if (false) {
|
|
/** @type {?|undefined} */
|
|
FcItemInfo.prototype.node;
|
|
/** @type {?|undefined} */
|
|
FcItemInfo.prototype.edge;
|
|
}
|
|
/**
|
|
* @record
|
|
*/
|
|
export function FcModel() { }
|
|
if (false) {
|
|
/** @type {?} */
|
|
FcModel.prototype.nodes;
|
|
/** @type {?} */
|
|
FcModel.prototype.edges;
|
|
}
|
|
/**
|
|
* @record
|
|
*/
|
|
export function UserCallbacks() { }
|
|
if (false) {
|
|
/** @type {?|undefined} */
|
|
UserCallbacks.prototype.dropNode;
|
|
/** @type {?|undefined} */
|
|
UserCallbacks.prototype.createEdge;
|
|
/** @type {?|undefined} */
|
|
UserCallbacks.prototype.edgeAdded;
|
|
/** @type {?|undefined} */
|
|
UserCallbacks.prototype.nodeRemoved;
|
|
/** @type {?|undefined} */
|
|
UserCallbacks.prototype.edgeRemoved;
|
|
/** @type {?|undefined} */
|
|
UserCallbacks.prototype.edgeDoubleClick;
|
|
/** @type {?|undefined} */
|
|
UserCallbacks.prototype.edgeMouseOver;
|
|
/** @type {?|undefined} */
|
|
UserCallbacks.prototype.isValidEdge;
|
|
/** @type {?|undefined} */
|
|
UserCallbacks.prototype.edgeEdit;
|
|
/** @type {?|undefined} */
|
|
UserCallbacks.prototype.nodeCallbacks;
|
|
}
|
|
/**
|
|
* @record
|
|
*/
|
|
export function UserNodeCallbacks() { }
|
|
if (false) {
|
|
/** @type {?|undefined} */
|
|
UserNodeCallbacks.prototype.nodeEdit;
|
|
/** @type {?|undefined} */
|
|
UserNodeCallbacks.prototype.doubleClick;
|
|
/** @type {?|undefined} */
|
|
UserNodeCallbacks.prototype.mouseDown;
|
|
/** @type {?|undefined} */
|
|
UserNodeCallbacks.prototype.mouseEnter;
|
|
/** @type {?|undefined} */
|
|
UserNodeCallbacks.prototype.mouseLeave;
|
|
}
|
|
/**
|
|
* @record
|
|
*/
|
|
export function FcCallbacks() { }
|
|
if (false) {
|
|
/** @type {?} */
|
|
FcCallbacks.prototype.nodeDragstart;
|
|
/** @type {?} */
|
|
FcCallbacks.prototype.nodeDragend;
|
|
/** @type {?} */
|
|
FcCallbacks.prototype.edgeDragstart;
|
|
/** @type {?} */
|
|
FcCallbacks.prototype.edgeDragend;
|
|
/** @type {?} */
|
|
FcCallbacks.prototype.edgeDrop;
|
|
/** @type {?} */
|
|
FcCallbacks.prototype.edgeDragoverConnector;
|
|
/** @type {?} */
|
|
FcCallbacks.prototype.edgeDragoverMagnet;
|
|
/** @type {?} */
|
|
FcCallbacks.prototype.edgeDragleaveMagnet;
|
|
/** @type {?} */
|
|
FcCallbacks.prototype.nodeMouseOver;
|
|
/** @type {?} */
|
|
FcCallbacks.prototype.nodeMouseOut;
|
|
/** @type {?} */
|
|
FcCallbacks.prototype.connectorMouseEnter;
|
|
/** @type {?} */
|
|
FcCallbacks.prototype.connectorMouseLeave;
|
|
/** @type {?} */
|
|
FcCallbacks.prototype.nodeClicked;
|
|
}
|
|
/**
|
|
* @record
|
|
*/
|
|
export function FcAdjacentList() { }
|
|
var BaseError = /** @class */ (function () {
|
|
function BaseError() {
|
|
Error.apply(this, arguments);
|
|
}
|
|
return BaseError;
|
|
}());
|
|
Object.defineProperty(BaseError, 'prototype', new Error());
|
|
var ModelvalidationError = /** @class */ (function (_super) {
|
|
tslib_1.__extends(ModelvalidationError, _super);
|
|
function ModelvalidationError(message) {
|
|
var _this = _super.call(this) || this;
|
|
_this.message = message;
|
|
return _this;
|
|
}
|
|
return ModelvalidationError;
|
|
}(BaseError));
|
|
export { ModelvalidationError };
|
|
if (false) {
|
|
/** @type {?} */
|
|
ModelvalidationError.prototype.message;
|
|
}
|
|
/**
|
|
* @param {?} graph
|
|
* @return {?}
|
|
*/
|
|
export function fcTopSort(graph) {
|
|
var e_1, _a, e_2, _b;
|
|
/** @type {?} */
|
|
var adjacentList = {};
|
|
graph.nodes.forEach((/**
|
|
* @param {?} node
|
|
* @return {?}
|
|
*/
|
|
function (node) {
|
|
adjacentList[node.id] = { incoming: 0, outgoing: [] };
|
|
}));
|
|
graph.edges.forEach((/**
|
|
* @param {?} edge
|
|
* @return {?}
|
|
*/
|
|
function (edge) {
|
|
/** @type {?} */
|
|
var sourceNode = graph.nodes.filter((/**
|
|
* @param {?} node
|
|
* @return {?}
|
|
*/
|
|
function (node) {
|
|
return node.connectors.some((/**
|
|
* @param {?} connector
|
|
* @return {?}
|
|
*/
|
|
function (connector) {
|
|
return connector.id === edge.source;
|
|
}));
|
|
}))[0];
|
|
/** @type {?} */
|
|
var destinationNode = graph.nodes.filter((/**
|
|
* @param {?} node
|
|
* @return {?}
|
|
*/
|
|
function (node) {
|
|
return node.connectors.some((/**
|
|
* @param {?} connector
|
|
* @return {?}
|
|
*/
|
|
function (connector) {
|
|
return connector.id === edge.destination;
|
|
}));
|
|
}))[0];
|
|
adjacentList[sourceNode.id].outgoing.push(destinationNode.id);
|
|
adjacentList[destinationNode.id].incoming++;
|
|
}));
|
|
/** @type {?} */
|
|
var orderedNodes = [];
|
|
/** @type {?} */
|
|
var sourceNodes = [];
|
|
try {
|
|
for (var _c = tslib_1.__values(Object.keys(adjacentList)), _d = _c.next(); !_d.done; _d = _c.next()) {
|
|
var node = _d.value;
|
|
/** @type {?} */
|
|
var edges = adjacentList[node];
|
|
if (edges.incoming === 0) {
|
|
sourceNodes.push(node);
|
|
}
|
|
}
|
|
}
|
|
catch (e_1_1) { e_1 = { error: e_1_1 }; }
|
|
finally {
|
|
try {
|
|
if (_d && !_d.done && (_a = _c.return)) _a.call(_c);
|
|
}
|
|
finally { if (e_1) throw e_1.error; }
|
|
}
|
|
while (sourceNodes.length !== 0) {
|
|
/** @type {?} */
|
|
var sourceNode = sourceNodes.pop();
|
|
for (var i = 0; i < adjacentList[sourceNode].outgoing.length; i++) {
|
|
/** @type {?} */
|
|
var destinationNode = adjacentList[sourceNode].outgoing[i];
|
|
adjacentList[destinationNode].incoming--;
|
|
if (adjacentList[destinationNode].incoming === 0) {
|
|
sourceNodes.push(destinationNode);
|
|
}
|
|
adjacentList[sourceNode].outgoing.splice(i, 1);
|
|
i--;
|
|
}
|
|
orderedNodes.push(sourceNode);
|
|
}
|
|
/** @type {?} */
|
|
var hasEdges = false;
|
|
try {
|
|
for (var _e = tslib_1.__values(Object.keys(adjacentList)), _f = _e.next(); !_f.done; _f = _e.next()) {
|
|
var node = _f.value;
|
|
/** @type {?} */
|
|
var edges = adjacentList[node];
|
|
if (edges.incoming !== 0) {
|
|
hasEdges = true;
|
|
}
|
|
}
|
|
}
|
|
catch (e_2_1) { e_2 = { error: e_2_1 }; }
|
|
finally {
|
|
try {
|
|
if (_f && !_f.done && (_b = _e.return)) _b.call(_e);
|
|
}
|
|
finally { if (e_2) throw e_2.error; }
|
|
}
|
|
if (hasEdges) {
|
|
return null;
|
|
}
|
|
else {
|
|
return orderedNodes;
|
|
}
|
|
}
|
|
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ngx-flowchart.models.js","sourceRoot":"ng://ngx-flowchart/","sources":["lib/ngx-flowchart.models.ts"],"names":[],"mappings":"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","sourcesContent":["import { Observable } from 'rxjs';\nimport { InjectionToken, Type } from '@angular/core';\nimport { FcNodeComponent } from './node.component';\n\nexport const FC_NODE_COMPONENT_CONFIG = new InjectionToken<FcNodeComponentConfig>('fc-node.component.config');\n\nexport interface FcNodeComponentConfig {\n  nodeComponentType: Type<FcNodeComponent>;\n}\n\nconst htmlPrefix = 'fc';\nconst leftConnectorType = 'leftConnector';\nconst rightConnectorType = 'rightConnector';\n\nexport const FlowchartConstants = {\n  htmlPrefix,\n  leftConnectorType,\n  rightConnectorType,\n  curvedStyle: 'curved',\n  lineStyle: 'line',\n  dragAnimationRepaint: 'repaint',\n  dragAnimationShadow: 'shadow',\n  canvasClass: htmlPrefix + '-canvas',\n  selectedClass: htmlPrefix + '-selected',\n  editClass: htmlPrefix + '-edit',\n  activeClass: htmlPrefix + '-active',\n  hoverClass: htmlPrefix + '-hover',\n  draggingClass: htmlPrefix + '-dragging',\n  edgeClass: htmlPrefix + '-edge',\n  edgeLabelClass: htmlPrefix + '-edge-label',\n  connectorClass: htmlPrefix + '-connector',\n  magnetClass: htmlPrefix + '-magnet',\n  nodeClass: htmlPrefix + '-node',\n  nodeOverlayClass: htmlPrefix + '-node-overlay',\n  leftConnectorClass: htmlPrefix + '-' + leftConnectorType + 's',\n  rightConnectorClass: htmlPrefix + '-' + rightConnectorType + 's',\n  canvasResizeThreshold: 200,\n  canvasResizeStep: 200\n};\n\n\nexport interface FcCoords {\n  x?: number;\n  y?: number;\n}\n\nexport interface FcRectBox {\n  top: number;\n  left: number;\n  right: number;\n  bottom: number;\n}\n\nexport interface FcConnector {\n  id: string;\n  type: string;\n}\n\nexport interface FcNode extends FcCoords {\n  id: string;\n  name: string;\n  connectors: Array<FcConnector>;\n  readonly?: boolean;\n  [key: string]: any;\n}\n\nexport interface FcEdge {\n  label?: string;\n  source?: string;\n  destination?: string;\n  active?: boolean;\n}\n\nexport interface FcItemInfo {\n  node?: FcNode;\n  edge?: FcEdge;\n}\n\nexport interface FcModel {\n  nodes: Array<FcNode>;\n  edges: Array<FcEdge>;\n}\n\nexport interface UserCallbacks {\n  dropNode?: (event: Event, node: FcNode) => void;\n  createEdge?: (event: Event, edge: FcEdge) => Observable<FcEdge>;\n  edgeAdded?: (edge: FcEdge) => void;\n  nodeRemoved?: (node: FcNode) => void;\n  edgeRemoved?: (edge: FcEdge) => void;\n  edgeDoubleClick?: (event: MouseEvent, edge: FcEdge) => void;\n  edgeMouseOver?: (event: MouseEvent, edge: FcEdge) => void;\n  isValidEdge?: (source: FcConnector, destination: FcConnector) => boolean;\n  edgeEdit?: (event: Event, edge: FcEdge) => void;\n  nodeCallbacks?: UserNodeCallbacks;\n}\n\nexport interface UserNodeCallbacks {\n  nodeEdit?: (event: MouseEvent, node: FcNode) => void;\n  doubleClick?: (event: MouseEvent, node: FcNode) => void;\n  mouseDown?: (event: MouseEvent, node: FcNode) => void;\n  mouseEnter?: (event: MouseEvent, node: FcNode) => void;\n  mouseLeave?: (event: MouseEvent, node: FcNode) => void;\n}\n\nexport interface FcCallbacks {\n  nodeDragstart: (event: DragEvent, node: FcNode) => void;\n  nodeDragend: (event: DragEvent) => void;\n  edgeDragstart: (event: DragEvent, connector: FcConnector) => void;\n  edgeDragend: (event: DragEvent) => void;\n  edgeDrop: (event: DragEvent, targetConnector: FcConnector) => boolean;\n  edgeDragoverConnector: (event: DragEvent, connector: FcConnector) => boolean;\n  edgeDragoverMagnet: (event: DragEvent, connector: FcConnector) => boolean;\n  edgeDragleaveMagnet: (event: DragEvent) => void;\n  nodeMouseOver: (event: MouseEvent, node: FcNode) => void;\n  nodeMouseOut: (event: MouseEvent, node: FcNode) => void;\n  connectorMouseEnter: (event: MouseEvent, connector: FcConnector) => void;\n  connectorMouseLeave: (event: MouseEvent, connector: FcConnector) => void;\n  nodeClicked: (event: MouseEvent, node: FcNode) => void;\n}\n\nexport interface FcAdjacentList {\n  [id: string]: {\n    incoming: number;\n    outgoing: Array<string>;\n  };\n}\n\nclass BaseError {\n  constructor() {\n    Error.apply(this, arguments);\n  }\n}\n\nObject.defineProperty(BaseError, 'prototype', new Error());\n\nexport class ModelvalidationError extends BaseError {\n  constructor(public message: string) {\n    super();\n  }\n}\n\nexport function fcTopSort(graph: FcModel): Array<string> | null {\n  const adjacentList: FcAdjacentList = {};\n  graph.nodes.forEach((node) => {\n    adjacentList[node.id] = {incoming: 0, outgoing: []};\n  });\n  graph.edges.forEach((edge) => {\n    const sourceNode = graph.nodes.filter((node) => {\n      return node.connectors.some((connector) => {\n        return connector.id === edge.source;\n      });\n    })[0];\n    const destinationNode = graph.nodes.filter((node) => {\n      return node.connectors.some((connector) => {\n        return connector.id === edge.destination;\n      });\n    })[0];\n    adjacentList[sourceNode.id].outgoing.push(destinationNode.id);\n    adjacentList[destinationNode.id].incoming++;\n  });\n  const orderedNodes: string[] = [];\n  const sourceNodes: string[] = [];\n  for (const node of Object.keys(adjacentList)) {\n    const edges = adjacentList[node];\n    if (edges.incoming === 0) {\n      sourceNodes.push(node);\n    }\n  }\n  while (sourceNodes.length !== 0) {\n    const sourceNode = sourceNodes.pop();\n    for (let i = 0; i < adjacentList[sourceNode].outgoing.length; i++) {\n      const destinationNode = adjacentList[sourceNode].outgoing[i];\n      adjacentList[destinationNode].incoming--;\n      if (adjacentList[destinationNode].incoming === 0) {\n        sourceNodes.push(destinationNode);\n      }\n      adjacentList[sourceNode].outgoing.splice(i, 1);\n      i--;\n    }\n    orderedNodes.push(sourceNode);\n  }\n  let hasEdges = false;\n  for (const node of Object.keys(adjacentList)) {\n    const edges = adjacentList[node];\n    if (edges.incoming !== 0) {\n      hasEdges = true;\n    }\n  }\n  if (hasEdges) {\n    return null;\n  } else {\n    return orderedNodes;\n  }\n}\n"]}
|