如何在使用GraphStream(或其他库)扩展JPanel的自定义java swing组件上绘制节点和边



我正在为java swing程序创建一个用户界面,该程序应该能够允许用户创建一个由边连接的节点网络。允许用户标记这些节点和边。

UI目前有一个扩展JPanel的自定义图形组件,它本身是有意义的,具有缩放和拖动等工作操作。我试图使用GraphStream库(http://graphstream-project.org/)将此自定义面板转换为支持GraphStream Graph s的View。由于GraphStream附带了一个DefaultView类,它能够开箱显示基本图形并扩展JPanel,所以我决定修改我的地图组件,以便它扩展DefaultView。所以背景组件的类定义现在是

public class CustomPanel extends DefaultView implements ICustomListener
{
    public CustomPanel(Viewer viewer, String identifier, GraphRenderer renderer)
    {
        super(viewer, identifier, renderer);
        ...
    }
}

但是,该组件无法显示任何图形或绘图功能。这就是我如何形成一个测试图

Graph graph = new GraphicGraph("embedded");
Viewer viewer = new Viewer(graph, Viewer.ThreadingModel.GRAPH_IN_GUI_THREAD);
Node a = graph.addNode("A");
a.addAttribute("xy", 0, 0);
Node b = graph.addNode("B");
b.addAttribute("xy", 10, 0);
Node c = graph.addNode("C");
c.addAttribute("xy", 10, 10);
graph.addEdge("AB","A","B");
//This is where I assume the GraphStream magic should happen.
CustomPanel myPanel = new CustomPanel(viewer, "defaultView", Viewer.newGraphRenderer());
viewer.addView(myPanel);
myPanel.display((GraphicGraph)(graph), true);

我可以用

显示相同的图形
viewer.addDefaultView(true);

但是它在一个新窗口中打开。

那么我做错了什么,或者甚至有可能在自定义组件上显示GraphStream图形?如果有其他图书馆可以做到这一点,我会很感激知道。我知道JUNG2和Java2D,但还没有尝试过。我也可以使用在另一个窗口中打开编辑器的解决方案,但我仍然需要我的CustomPanel作为背景。

如果有人面临同样的问题,下面是我克服它的方法。请记住,保持后台组件的所有原始功能至关重要。总的来说,GraphStream的DefaultView的实现是一个很好的例子。

我做了一个类,从后台扩展和实现org.graphstream.ui.view.View

public class HybridView extends CustomView implements View

然后我发现我必须修改默认的GraphStream渲染器,其中编写自己的renderGraph方法至关重要。我只是从超类中复制了实现,并注释掉了一些行:

protected void renderGraph(Graphics2D g) {
    StyleGroup style = graph.getStyle();
    setupGraphics(g);
    //Dont paint on the background
    //renderGraphBackground(g);
    //Commenting out the next line had no visible effects so I left it unmodified, tho it's name begs for being commented out
    renderBackLayer(g);
    camera.pushView(graph, g);
    renderGraphElements(g);
    if (style.getStrokeMode() != StyleConstants.StrokeMode.NONE
            && style.getStrokeWidth().value != 0) {
        GraphMetrics metrics = camera.getMetrics();
        Rectangle2D rect = new Rectangle2D.Double();
        double px1 = metrics.px1;
        Value stroke = style.getShadowWidth();
        rect.setFrame(metrics.lo.x, metrics.lo.y + px1,
                metrics.size.data[0] - px1, metrics.size.data[1] - px1);
        g.setStroke(new BasicStroke((float) metrics.lengthToGu(stroke)));
        g.setColor(graph.getStyle().getStrokeColor(0));
        g.draw(rect);
    }
    camera.popView(g);
    renderForeLayer(g);
}
public class MyGraphRenderer extends GraphRendererBase 
然后使用GraphStream 的绘图功能的组件我以前存在的有意义的背景组件:
Graph graph = new GraphicGraph("embedded");
Viewer viewer = new Viewer(graph, Viewer.ThreadingModel.GRAPH_IN_GUI_THREAD);
CustomPanel myPanel = new CustomPanel(viewer, "defaultView", new MyGraphRenderer());

我不知道你想做什么。但是如果你只是想在你自己的GUI中添加一个GraphStream视图,那么你不需要扩展DefaultView。您创建的查看器是OK的,因此您只需要在GUI中插入一个新视图,如下所示:

    // create a view *without* a JFrame
    View view = viewer.addDefaultView(false);
    view.setPreferredSize(new Dimension(600, 400));
    JFrame f = new JFrame("Your GUI");
    // add the view to your components (JFrame, JPanel...)
    f.add(view, BorderLayout.CENTER);
    // add your other GUI stuff here...
    f.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
    f.pack();
    f.setVisible(true);

最新更新