Cytoscape将自定义图像添加到节点上,并在其上显示可点击的项目

时间:2017-12-07 19:37:38

标签: javascript ember.js cytoscape.js

我正在使用cytoscape js在ember中构建一个图表工具,我可以渲染图表数据但是我不知道如何将每个节点设置为显示其中包含其他图像/按钮的图像。基本上我希望它看起来像这样:

enter image description here

在图像中有两个按钮(我很可能也会添加图标),节点中也存在标签,我不知道该怎么做。

以下是我目前的代码。

模板:

<div class="container" >
  <div id="cy"></div>
</div>

组件JS:

import Ember from 'ember';


export default Ember.Component.extend({
tagName: '',

map: Ember.computed('model.map_data', function()
{
 if(this.get('model.map_data')){
   return JSON.parse(this.get('model.map_data').data)
  } else {
   return {};
  }
 }),
cytoscape_data: Ember.computed('model.sub_apps.[]',function() {
var ret = {
        nodes: [],
        edges: []
};
var red = 50;//replace with threshold
var green = 25;//replace with threshold
var _this = this;
this.get("model").map_data.forEach(function(node) {
  var y= 0;
  var x = 0;
  var color = 'green';
  if(node.value >= red ){
    color = 'red';
  }else {
    if(node.value > green){
      color = 'orange';
    }
  }
  var position = _this.get("map")["app" + node.id];
  if(position){
    x = parseInt(position.split(',')[0]);
    y = parseInt(position.split(',')[1]);
  }
  ret["nodes"].push({
          data: {
                  id: node.id,
                  label: node.name,
                  node_type: 'app',
                  tooltip: node.description,
                  color: color
          },
          position: {
                  x: x,
                  y: y
          }
  });
  if(node.relations) {
    node.relations.forEach(function(parent) {

      ret["edges"].push({
        data: {
          source: node.id,
          target: parent.app_to_id
        }
      });
    });
  }
});

 return ret;
}),

didInsertElement: function() {
 this._super();
 var cy = cytoscape({
 container: Ember.$('#cy')[0],
 elements: this.get("cytoscape_data"),
 zoom: 1,
 pan: { x: 0, y: 0 },
 fit: true,
 randomize: false,
 layout: {
      name: 'preset'
    },
  style: [
    {
      selector: 'node',
      style: {
        'content': 'data(label)',
        'text-opacity': 0.8,
        'text-valign': 'center',
        'text-halign': 'right',
        'width': '200px',
        'height': '200px',
        'border-color': 'green',
        'border-width': 3,
        'border-opacity': 0.5,
        'background-image': 'url(../assets/images/base_node_image.svg)'
        // 'background-color': 'data(color)'
      }
    },
    {
      selector: 'edge',
      style: {
        'width': 6,
        'border-color': 'green',
        'target-arrow-shape': 'triangle',
        'target-arrow-color': 'red',
        'opacity': 1,
        'curve-style': 'bezier'
      }
    },

    {
      selector: ':selected',
      style: {
        'background-color': 'orange',
        'opacity': 1
      }
    },

    {
      selector: '.faded',
      style: {
        'opacity': 0.0,
        'text-opacity': 0
      }
    },
  ],

});
Ember.run.scheduleOnce('afterRender', this, function(){
  cy;
});
cy.on('click', 'node', function(evt){
  var node = evt.target;
  console.log( 'clicked ' + node.data('label') );
   });
  },
});

此代码呈现的图表如下所示:

enter image description here

我可以显示背景图像,但它显示在一个我不知道如何摆脱的圆圈中。圆圈的颜色由上面的一些逻辑确定,该逻辑是一个测试,看它是否有效,并且很好(稍后将用于节点上的一个图标)。我也可以显示节点的标签,但我不知道如何在节点内显示它。

感谢任何帮助,谢谢!

1 个答案:

答案 0 :(得分:1)

如果不是不可能的话,实现你想要的东西并不是那么简单。你说“设置每个节点显示的图像有其​​他图像/按钮在其中运作。”;这意味着你需要将html渲染到画布中;因为cytoscape作为绘图区域的内容是HTML canvas

请参阅@ maxfranz(cytoscape.js的作者)relevant question;在那里他基本上说“不可能在画布上呈现HTML,也不可能想要表现”。

这意味着放置html按钮,URL可能不是你想要的。另请参阅MDN Web Docs以获得进一步说明。

那说;我认为你仍然可以实现你想要的目标;但采用不同的方法。您可以使用cytoscape的{​​{1}} s。您可以将图像和按钮定义为简单节点,并将复合节点定义为周围容器。我已在follwoing github repository为您创建了一个工作示例。

我得到的最终结果如下:

enter image description here

我希望这会有所帮助。