为什么我的嵌入式JointJS元素重叠?

时间:2016-03-30 16:19:38

标签: javascript jointjs

我正在使用JointJS图表,使用DirectedGraph来处理布局,我正在尝试实现类似于下图的内容。我需要将节点(A,B,C,D,E,F,G,H,I,J)“概括”或包含在单独的节点(Foo,Bar,Hmm)中。当我将所有元素添加到图表中时,一切都在彼此之上。但是,如果我不在区域之间添加顶点,则所有元素都会正确布局,但没有连接区域的顶点。

根据下面的代码,我做错了什么?节点(A,B,C,D,E,F,G,H,I,J)是否会导致错误,因为它们没有连接到图表的其余部分?

非常感谢您提供的任何反馈。

enter image description here

var graph = new joint.dia.Graph;
var paper = new joint.dia.Paper({
    el: $('#paper');
    width: 2000,
    height: 2000,
    gridSize: 1,
    model: graph
});

var regions = [ makeRegion('Foo'), makeRegion('Bar'), makeRegion('Hmm')];
var nodes = [
    makeNode('A'),
    makeNode('B'),
    makeNode('C'),
    makeNode('D'),
    makeNode('F'),
    makeNode('G'),
    makeNode('H'),
    makeNode('I'),
    makeNode('J'),
];
regions[0].embed(nodes[0]).embed(nodes[1]).embed(nodes[2]);
regions[1].embed(nodes[3]).embed(nodes[4]).embed(nodes[5]);
regions[2].embed(nodes[6]).embed(nodes[7]).embed(nodes[8]);
var vertices = [
    connect(regions[0], regions[1]),
    connect(regions[1], regions[2]),
    connect(regions[2], regions[0])
    ];
paper.addCells(regions.concat(nodes).concat(vertices));
joint.layout.DirectedGraph.layout(paper, {
    rankDir: 'LR',
    marginX: 30,
    marginY: 30,
    clusterPadding: {
        top: 30,
        left: 10,
        right: 10,
        bottom: 10
    }
});

function makeNode(name) {
    return new joint.shapes.basic.Rect({
        size: {
            width: 35,
            height: 35
        },
        attrs: {
            text: {
                text: name
            }
        }
    });
}

function connect(a, b) {
    return new joint.shapes.fsa.Arrow({
        source: { id: a.id },
        target: { id: b.id }
    });
}

function makeRegion(name) {
    return new joint.shapes.basic.Rect({
        size: {
            width: 300,
            height: 200
        },
        attrs: {
            text: {
                text: name
            }
        }
    });
}

编辑:

虽然我从来没有找到解决这个问题的好方法,但我确实发现,通过一些额外的工作,这可以实现。它假设您的图形不是太复杂,区域之间的边缘很少。

如果我单独添加边缘,在将所有节点添加到图形之后,它将正常工作。然而,定向布局并不完全按预期工作。我最初使用joint.layout.DirectedGraph.layout()以便它将对齐内部节点,然后重新定位out区域。最后我添加了边缘,并产生类似于上图的结果。

以下是我如何实现这一目标的大致概述:

var graph = new joint.dia.Graph;
var paper = new joint.dia.Paper({
    el: $('#paper');
    width: 2000,
    height: 2000,
    gridSize: 1,
    model: graph
});

var regions = [ makeRegion('Foo'), makeRegion('Bar'), makeRegion('Hmm')];
var nodes = [
    makeNode('A'),
    makeNode('B'),
    makeNode('C'),
    makeNode('D'),
    makeNode('F'),
    makeNode('G'),
    makeNode('H'),
    makeNode('I'),
    makeNode('J'),
];
regions[0].embed(nodes[0]).embed(nodes[1]).embed(nodes[2]);
regions[1].embed(nodes[3]).embed(nodes[4]).embed(nodes[5]);
regions[2].embed(nodes[6]).embed(nodes[7]).embed(nodes[8]);

paper.addCells(regions.concat(nodes));
joint.layout.DirectedGraph.layout(paper, {
    rankDir: 'LR',
    marginX: 30,
    marginY: 30,
    clusterPadding: {
        top: 30,
        left: 10,
        right: 10,
        bottom: 10
    }
});

var vertices = [
    connect(regions[0], regions[1]),
    connect(regions[1], regions[2]),
    connect(regions[2], regions[0])
    ];
repositionRegions();
// Add the edges after all the nodes are in their final position
graph.addCells(vertices);

function repositionRegions() {
    // Move regions to where you want them
}

function makeNode(name) {
    return new joint.shapes.basic.Rect({
        size: {
            width: 35,
            height: 35
        },
        attrs: {
            text: {
                text: name
            }
        }
    });
}

function connect(a, b) {
    return new joint.shapes.fsa.Arrow({
        source: { id: a.id },
        target: { id: b.id }
    });
}

function makeRegion(name) {
    return new joint.shapes.basic.Rect({
        size: {
            width: 300,
            height: 200
        },
        attrs: {
            text: {
                text: name
            }
        }
    });
}

1 个答案:

答案 0 :(得分:0)

这是Dagre-D3的已知问题:Automatic layout does not work on hierarchical diagrams with links with parent

作为一种解决方法,您可以省略区域之间的链接(在您的问题的版本中建议),或者可以做一些额外的工作来很好地处理任意数量的节点:

  1. 分别布置每组子节点:

    var fooChildren = [nodes[0], nodes[1], nodes[2]];
    var barChildren = [nodes[3], nodes[4], nodes[5]];
    var hmmChildren = [nodes[6], nodes[7], nodes[8]];
    var children = [fooChildren, barChildren, hmmChildren];
    
    for(var i = 0; i < children.length; i++)
        joint.layout.DirectedGraph.layout(children[i]);
    
  2. 为每个区域创建一个辅助克隆区域。

    var clones = [];
    for(var i = 0; i < regions.length; i++) {
        var clone = regions[i].clone();
        graph.addCell(clone);
        clones.push(clone);
    }
    
  3. 将相应的节点设置为其克隆区域。

    for(var i = 0; i < children.length; i ++)
        for(var k = 0; k < children[i].length; k++)
            clones[i].embed(children[i][k]);
    
  4. 将每个克隆的区域调整为适合其子区域的大小,并使用克隆的区域的大小调整原始区域的大小。

    for(var i = 0; i < clones.length; i++) {
        clones[i].fitEmbeds(padding: { top: 30, left: 10, right: 10, bottom: 10 });
        regions[i].resize(clones[i].getBBox().width, clones[i].getBBox().height);
    }
    
  5. 布置图。

    joint.layout.DirectedGraph.layout(graph, {
        rankDir: 'LR',
        marginX: 30,
        marginY: 30,
        clusterPadding: {
            top: 30,
            left: 10,
            right: 10,
            bottom: 10
        }
    });
    
  6. 将克隆的区域翻译到其原始区域的位置(子代将相应翻译)。

    for(var i = 0; i < clones.length; i++) {
        var dx = regions[i].getBBox().x - clones[i].getBBox().x;
        var dy = regions[i].getBBox().y - clones[i].getBBox().y;
        clones[i].translate(dx, dy);
    }
    
  7. 删除克隆的区域,并将子区域设置为原始区域。

    for(var i = 0; i < regions.length; i++) {
        clones[i].remove();
    
        for(var j = 0; j < children[i].length; j++)
            regions[i].embed(children[i][j]);
    }
    

希望这可以帮助提供替代解决方案。