cytoscape-angular
Version:
Angular 12+ components for cytoscape charting.
231 lines • 23.7 kB
JavaScript
import { Component, Input, ViewChild } from '@angular/core';
import * as i0 from "@angular/core";
import * as i1 from "@angular/common";
import * as i2 from "primeng/progressspinner";
const _c0 = ["cyGraph"];
function CytoscapeGraphComponent_p_progressSpinner_0_Template(rf, ctx) { if (rf & 1) {
i0.ɵɵelement(0, "p-progressSpinner", 3);
} }
/**
* The API is a little odd to provide flexibility.
* EITHER bind to cyOptions (type CytoscapeOptions), to control the options yourself
* OR this component will build a CytoscapeOptions internally by using all the other inputs.
* If cyOptions is supplied, all other inputs are ignored.
* The cyOptions container (HTML element) is always ignored and set internally.
*/
export class CytoscapeGraphComponent {
constructor() {
this.debug = false;
this.showToolbar = true;
this.loading = false;
}
ngOnChanges(changes) {
console.log('cytoscape graph component ngOnChanges. changes:', JSON.stringify(changes));
if (changes["style"]) {
console.log('changes["style"]:', JSON.stringify(changes["style"]));
this.runWhileLoading(this.updateStyles.bind(this));
}
}
centerElements(selector) {
if (!this.cy) {
return;
}
const elems = this.cy.$(selector);
this.cy.center(elems);
}
zoomToElement(selector, level = 3) {
var _a, _b;
let position = (_b = (_a = this.cy) === null || _a === void 0 ? void 0 : _a.$(selector)) === null || _b === void 0 ? void 0 : _b.position();
if (!position) {
console.warn(`Cannot zoom to ${selector}`);
}
this.cy.zoom({
level: level,
position: position
});
}
render() {
this.runWhileLoading(this.rerender.bind(this));
}
runWhileLoading(f) {
this.loading = true;
setTimeout(() => {
f();
setTimeout(() => {
this.loading = false;
}, 30);
}, 0);
}
updateStyles() {
if (this.cy && this.style) {
this.cy.style(this.style);
}
}
rerender() {
//TODO : this takes a heavy-handed approach, refine for performance
if (!this.cyGraph) {
console.warn(`No cyGraph found`);
return;
}
const cyOptions = this.cyOptions || {
// ignored, use nodes and edges
// elements: this.elements,
autolock: this.autolock,
autoungrabify: this.autoungrabify,
autounselectify: this.autounselectify,
boxSelectionEnabled: this.boxSelectionEnabled,
container: this.cyGraph.nativeElement,
desktopTapThreshold: this.desktopTapThreshold,
hideEdgesOnViewport: this.hideEdgesOnViewport,
hideLabelsOnViewport: this.hideLabelsOnViewport,
layout: this.layoutOptions,
maxZoom: this.maxZoom,
minZoom: this.minZoom,
motionBlur: this.motionBlur,
motionBlurOpacity: this.motionBlurOpacity,
pan: this.pan,
panningEnabled: this.panningEnabled,
pixelRatio: this.pixelRatio,
selectionType: this.selectionType,
style: this.style,
styleEnabled: this.styleEnabled,
textureOnViewport: this.textureOnViewport,
touchTapThreshold: this.touchTapThreshold,
userPanningEnabled: this.userPanningEnabled,
userZoomingEnabled: this.userZoomingEnabled,
wheelSensitivity: this.wheelSensitivity,
zoomingEnabled: this.zoomingEnabled,
zoom: this.zoom,
};
// TODO do reset() instead?
this.cy = cytoscape(cyOptions);
this.cy.startBatch();
this.cy.boxSelectionEnabled(this.boxSelectionEnabled);
this.cy.nodes().remove();
this.cy.edges().remove();
if (this.nodes) {
this.cy.add(this.nodes);
}
if (this.edges) {
this.cy.add(this.edges);
}
this.cy.endBatch();
if (this.layoutOptions) {
this.cy.layout(this.layoutOptions).run();
}
}
}
CytoscapeGraphComponent.ɵfac = function CytoscapeGraphComponent_Factory(t) { return new (t || CytoscapeGraphComponent)(); };
CytoscapeGraphComponent.ɵcmp = /*@__PURE__*/ i0.ɵɵdefineComponent({ type: CytoscapeGraphComponent, selectors: [["cytoscape-graph"]], viewQuery: function CytoscapeGraphComponent_Query(rf, ctx) { if (rf & 1) {
i0.ɵɵviewQuery(_c0, 5);
} if (rf & 2) {
let _t;
i0.ɵɵqueryRefresh(_t = i0.ɵɵloadQuery()) && (ctx.cyGraph = _t.first);
} }, inputs: { debug: "debug", nodes: "nodes", edges: "edges", autolock: "autolock", autoungrabify: "autoungrabify", autounselectify: "autounselectify", boxSelectionEnabled: "boxSelectionEnabled", desktopTapThreshold: "desktopTapThreshold", hideEdgesOnViewport: "hideEdgesOnViewport", hideLabelsOnViewport: "hideLabelsOnViewport", layoutOptions: "layoutOptions", maxZoom: "maxZoom", minZoom: "minZoom", motionBlur: "motionBlur", motionBlurOpacity: "motionBlurOpacity", pan: "pan", panningEnabled: "panningEnabled", pixelRatio: "pixelRatio", selectionType: "selectionType", style: "style", styleEnabled: "styleEnabled", textureOnViewport: "textureOnViewport", touchTapThreshold: "touchTapThreshold", userPanningEnabled: "userPanningEnabled", userZoomingEnabled: "userZoomingEnabled", wheelSensitivity: "wheelSensitivity", zoom: "zoom", zoomingEnabled: "zoomingEnabled", showToolbar: "showToolbar" }, features: [i0.ɵɵNgOnChangesFeature], decls: 3, vars: 1, consts: [["class", "spinner", "strokeWidth", "4", "fill", "#EEEEEE", "animationDuration", ".5s", 4, "ngIf"], [1, "graphWrapper"], ["cyGraph", ""], ["strokeWidth", "4", "fill", "#EEEEEE", "animationDuration", ".5s", 1, "spinner"]], template: function CytoscapeGraphComponent_Template(rf, ctx) { if (rf & 1) {
i0.ɵɵtemplate(0, CytoscapeGraphComponent_p_progressSpinner_0_Template, 1, 0, "p-progressSpinner", 0);
i0.ɵɵelement(1, "div", 1, 2);
} if (rf & 2) {
i0.ɵɵproperty("ngIf", ctx.loading);
} }, directives: [i1.NgIf, i2.ProgressSpinner], styles: [".spinner[_ngcontent-%COMP%] {\n position: absolute;\n left: '350px';\n z-index: 10;\n width: '250px';\n height: '250px';\n }\n @keyframes ui-progress-spinner-color {\n 100%,\n 0% {\n stroke: #d62d20;\n }\n 40% {\n stroke: #0057e7;\n }\n 66% {\n stroke: #008744;\n }\n 80%,\n 90% {\n stroke: #ffa700;\n }\n }\n .graphWrapper[_ngcontent-%COMP%] {\n height: 100%;\n width: 100%;\n }"] });
(function () { (typeof ngDevMode === "undefined" || ngDevMode) && i0.ɵsetClassMetadata(CytoscapeGraphComponent, [{
type: Component,
args: [{
selector: 'cytoscape-graph',
template: `
<p-progressSpinner *ngIf="loading" class="spinner" strokeWidth="4" fill="#EEEEEE" animationDuration=".5s"></p-progressSpinner>
<div #cyGraph class="graphWrapper">
</div>
`,
styles: [`
.spinner {
position: absolute;
left: '350px';
z-index: 10;
width: '250px';
height: '250px';
}
ui-progress-spinner-color {
100%,
0% {
stroke: #d62d20;
}
40% {
stroke: #0057e7;
}
66% {
stroke: #008744;
}
80%,
90% {
stroke: #ffa700;
}
}
.graphWrapper {
height: 100%;
width: 100%;
}`
]
}]
}], function () { return []; }, { cyGraph: [{
type: ViewChild,
args: ['cyGraph']
}], debug: [{
type: Input
}], nodes: [{
type: Input
}], edges: [{
type: Input
}], autolock: [{
type: Input
}], autoungrabify: [{
type: Input
}], autounselectify: [{
type: Input
}], boxSelectionEnabled: [{
type: Input
}], desktopTapThreshold: [{
type: Input
}], hideEdgesOnViewport: [{
type: Input
}], hideLabelsOnViewport: [{
type: Input
}], layoutOptions: [{
type: Input
}], maxZoom: [{
type: Input
}], minZoom: [{
type: Input
}], motionBlur: [{
type: Input
}], motionBlurOpacity: [{
type: Input
}], pan: [{
type: Input
}], panningEnabled: [{
type: Input
}], pixelRatio: [{
type: Input
}], selectionType: [{
type: Input
}], style: [{
type: Input
}], styleEnabled: [{
type: Input
}], textureOnViewport: [{
type: Input
}], touchTapThreshold: [{
type: Input
}], userPanningEnabled: [{
type: Input
}], userZoomingEnabled: [{
type: Input
}], wheelSensitivity: [{
type: Input
}], zoom: [{
type: Input
}], zoomingEnabled: [{
type: Input
}], showToolbar: [{
type: Input
}] }); })();
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cytoscape-graph.component.js","sourceRoot":"","sources":["../../../../projects/cytoscape-angular/src/lib/cytoscape-graph.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { AfterViewInit, Component, ElementRef, Input, OnChanges, SimpleChanges, ViewChild } from '@angular/core'\nimport * as cy from 'cytoscape'\nimport {\n  CytoscapeOptions,\n  EdgeDefinition,\n  LayoutOptions,\n  NodeDefinition,\n  Position,\n  SelectionType,\n  Stylesheet\n} from 'cytoscape'\n\ndeclare var cytoscape: any\n\n/**\n * The API is a little odd to provide flexibility.\n * EITHER bind to cyOptions (type CytoscapeOptions), to control the options yourself\n * OR this component will build a CytoscapeOptions internally by using all the other inputs.\n * If cyOptions is supplied, all other inputs are ignored.\n * The cyOptions container (HTML element) is always ignored and set internally.\n */\n@Component({\n  selector: 'cytoscape-graph',\n  template: `\n    <p-progressSpinner *ngIf=\"loading\" class=\"spinner\" strokeWidth=\"4\" fill=\"#EEEEEE\" animationDuration=\".5s\"></p-progressSpinner>\n    <div #cyGraph class=\"graphWrapper\">\n    </div>\n  `,\n  styles: [`\n    .spinner {\n      position: absolute;\n      left: '350px';\n      z-index: 10;\n      width: '250px';\n      height: '250px';\n    }\n    @keyframes ui-progress-spinner-color {\n      100%,\n      0% {\n        stroke: #d62d20;\n      }\n      40% {\n        stroke: #0057e7;\n      }\n      66% {\n        stroke: #008744;\n      }\n      80%,\n      90% {\n        stroke: #ffa700;\n      }\n    }\n    .graphWrapper {\n      height: 100%;\n      width: 100%;\n    }`\n  ]\n})\nexport class CytoscapeGraphComponent implements OnChanges {\n  @ViewChild('cyGraph')\n  cyGraph: ElementRef\n\n  @Input()\n  debug = false\n\n  @Input()\n  nodes: NodeDefinition[]\n  @Input()\n  edges: EdgeDefinition[]\n\n  @Input()\n  autolock: boolean\n  @Input()\n  autoungrabify: boolean\n  @Input()\n  autounselectify: boolean\n  @Input()\n  boxSelectionEnabled: boolean\n  @Input()\n  desktopTapThreshold: number\n  @Input()\n  hideEdgesOnViewport: boolean\n  @Input()\n  hideLabelsOnViewport: boolean\n  @Input()\n  layoutOptions: LayoutOptions\n  @Input()\n  maxZoom: number\n  @Input()\n  minZoom: number\n  @Input()\n  motionBlur: boolean\n  @Input()\n  motionBlurOpacity: number\n  @Input()\n  pan: Position\n  @Input()\n  panningEnabled: boolean\n  @Input()\n  pixelRatio: number | 'auto'\n  @Input()\n  selectionType: SelectionType\n  @Input()\n  style: Stylesheet[]\n  @Input()\n  styleEnabled: boolean\n  @Input()\n  textureOnViewport: boolean\n  @Input()\n  touchTapThreshold: number\n  @Input()\n  userPanningEnabled: boolean\n  @Input()\n  userZoomingEnabled: boolean\n  @Input()\n  wheelSensitivity: number\n  @Input()\n  zoom: 1\n  @Input()\n  zoomingEnabled: boolean\n  @Input()\n  showToolbar = true\n\n  cyOptions: CytoscapeOptions\n  private cy: cy.Core\n  loading: boolean = false\n\n  constructor() {\n  }\n\n  public ngOnChanges(changes: SimpleChanges): any {\n    console.log('cytoscape graph component ngOnChanges. changes:', JSON.stringify(changes))\n      if (changes[\"style\"]) {\n        console.log('changes[\"style\"]:', JSON.stringify(changes[\"style\"]))\n        this.runWhileLoading(this.updateStyles.bind(this))\n      }\n  }\n\n  public centerElements(selector) {\n    if (!this.cy) {\n      return\n    }\n    const elems = this.cy.$(selector)\n    this.cy.center(elems)\n  }\n\n  public zoomToElement(selector: string, level = 3) {\n    let position = this.cy?.$(selector)?.position()\n    if (!position) {\n      console.warn(`Cannot zoom to ${selector}`)\n    }\n    this.cy.zoom({\n      level: level,\n      position: position\n    });\n  }\n\n  public render() {\n    this.runWhileLoading(this.rerender.bind(this))\n  }\n\n  public runWhileLoading(f: Function) {\n    this.loading = true\n    setTimeout(()=> {\n      f()\n      setTimeout(() => {\n        this.loading = false\n      }, 30)\n    }, 0)\n  }\n\n  private updateStyles() {\n    if (this.cy && this.style) {\n      this.cy.style(this.style)\n    }\n  }\n\n  public rerender() {\n    //TODO : this takes a heavy-handed approach, refine for performance\n    if (!this.cyGraph) {\n      console.warn(`No cyGraph found`)\n      return\n    }\n\n    const cyOptions = this.cyOptions || {\n      // ignored, use nodes and edges\n      // elements: this.elements,\n      autolock: this.autolock,\n      autoungrabify: this.autoungrabify,\n      autounselectify: this.autounselectify,\n      boxSelectionEnabled: this.boxSelectionEnabled,\n      container: this.cyGraph.nativeElement,\n      desktopTapThreshold: this.desktopTapThreshold,\n      hideEdgesOnViewport: this.hideEdgesOnViewport,\n      hideLabelsOnViewport: this.hideLabelsOnViewport,\n      layout: this.layoutOptions,\n      maxZoom: this.maxZoom,\n      minZoom: this.minZoom,\n      motionBlur: this.motionBlur,\n      motionBlurOpacity: this.motionBlurOpacity,\n      pan: this.pan,\n      panningEnabled: this.panningEnabled,\n      pixelRatio: this.pixelRatio,\n      selectionType: this.selectionType,\n      style: this.style,\n      styleEnabled: this.styleEnabled,\n      textureOnViewport: this.textureOnViewport,\n      touchTapThreshold: this.touchTapThreshold,\n      userPanningEnabled: this.userPanningEnabled,\n      userZoomingEnabled: this.userZoomingEnabled,\n      wheelSensitivity: this.wheelSensitivity,\n      zoomingEnabled: this.zoomingEnabled,\n      zoom: this.zoom,\n    }\n    // TODO do reset() instead?\n    this.cy = cytoscape(cyOptions)\n    this.cy.startBatch()\n    this.cy.boxSelectionEnabled(this.boxSelectionEnabled)\n    this.cy.nodes().remove()\n    this.cy.edges().remove()\n    if (this.nodes) {\n      this.cy.add(this.nodes)\n    }\n    if (this.edges) {\n      this.cy.add(this.edges)\n    }\n    this.cy.endBatch()\n    if (this.layoutOptions) {\n      this.cy.layout(this.layoutOptions).run()\n    }\n  }\n}\n\n/*\nGradient:\n\nbackground-gradient-stop-colors : The colours of the background gradient stops (e.g. cyan magenta yellow).\nbackground-gradient-stop-positions : The positions of the background gradient stops (e.g. 0% 50% 100%). If not specified or invalid, the stops will divide equally.\nbackground-gradient-direction : For background-fill: linear-gradient, this property defines the direction of the background gradient. The following values are accepted:\nto-bottom (default)\nto-top\nto-left\nto-right\nto-bottom-right\nto-bottom-left\nto-top-right\nto-top-left\n */\n"]}