UNPKG

cytoscape-angular

Version:
231 lines 23.7 kB
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'; } @keyframes 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"]}