Force-directedレイアウトに新しいノードを追加する


89

Stack Overflowの最初の質問なので、我慢してください!私はd3.jsを初めて使用しましたが、他の人がそれを使って達成できることに一貫して驚かされています...そして、私が自分でどれほどの進歩を遂げることができたかに驚かされています!明らかに私は何かを壊していないので、私はここの親切な魂が私に光を見せてくれることを願っています。

私の意図は、単に以下を実行する再利用可能なJavaScript関数を作成することです。

  • 指定されたDOM要素に空の有向グラフを作成します
  • ラベル付けされた画像を含むノードをそのグラフに追加および削除し、それらの間の接続を指定できます。

私はhttp://bl.ocks.org/950642を出発点として採用しました。これは、本質的に、私が作成したい種類のレイアウトだからです。

ここに画像の説明を入力してください

私のコードは次のようになります:

<!DOCTYPE html>
<html>
<head>
    <script type="text/javascript" src="jquery.min.js"></script>
    <script type="text/javascript" src="underscore-min.js"></script>
    <script type="text/javascript" src="d3.v2.min.js"></script>
    <style type="text/css">
        .link { stroke: #ccc; }
        .nodetext { pointer-events: none; font: 10px sans-serif; }
        body { width:100%; height:100%; margin:none; padding:none; }
        #graph { width:500px;height:500px; border:3px solid black;border-radius:12px; margin:auto; }
    </style>
</head>
<body>
<div id="graph"></div>
</body>
<script type="text/javascript">

function myGraph(el) {

    // Initialise the graph object
    var graph = this.graph = {
        "nodes":[{"name":"Cause"},{"name":"Effect"}],
        "links":[{"source":0,"target":1}]
    };

    // Add and remove elements on the graph object
    this.addNode = function (name) {
        graph["nodes"].push({"name":name});
        update();
    }

    this.removeNode = function (name) {
        graph["nodes"] = _.filter(graph["nodes"], function(node) {return (node["name"] != name)});
        graph["links"] = _.filter(graph["links"], function(link) {return ((link["source"]["name"] != name)&&(link["target"]["name"] != name))});
        update();
    }

    var findNode = function (name) {
        for (var i in graph["nodes"]) if (graph["nodes"][i]["name"] === name) return graph["nodes"][i];
    }

    this.addLink = function (source, target) {
        graph["links"].push({"source":findNode(source),"target":findNode(target)});
        update();
    }

    // set up the D3 visualisation in the specified element
    var w = $(el).innerWidth(),
        h = $(el).innerHeight();

    var vis = d3.select(el).append("svg:svg")
        .attr("width", w)
        .attr("height", h);

    var force = d3.layout.force()
        .nodes(graph.nodes)
        .links(graph.links)
        .gravity(.05)
        .distance(100)
        .charge(-100)
        .size([w, h]);

    var update = function () {

        var link = vis.selectAll("line.link")
            .data(graph.links);

        link.enter().insert("line")
            .attr("class", "link")
            .attr("x1", function(d) { return d.source.x; })
            .attr("y1", function(d) { return d.source.y; })
            .attr("x2", function(d) { return d.target.x; })
            .attr("y2", function(d) { return d.target.y; });

        link.exit().remove();

        var node = vis.selectAll("g.node")
            .data(graph.nodes);

        node.enter().append("g")
            .attr("class", "node")
            .call(force.drag);

        node.append("image")
            .attr("class", "circle")
            .attr("xlink:href", "https://d3nwyuy0nl342s.cloudfront.net/images/icons/public.png")
            .attr("x", "-8px")
            .attr("y", "-8px")
            .attr("width", "16px")
            .attr("height", "16px");

        node.append("text")
            .attr("class", "nodetext")
            .attr("dx", 12)
            .attr("dy", ".35em")
            .text(function(d) { return d.name });

        node.exit().remove();

        force.on("tick", function() {
          link.attr("x1", function(d) { return d.source.x; })
              .attr("y1", function(d) { return d.source.y; })
              .attr("x2", function(d) { return d.target.x; })
              .attr("y2", function(d) { return d.target.y; });

          node.attr("transform", function(d) { return "translate(" + d.x + "," + d.y + ")"; });
        });

        // Restart the force layout.
        force
          .nodes(graph.nodes)
          .links(graph.links)
          .start();
    }

    // Make it all go
    update();
}

graph = new myGraph("#graph");

// These are the sort of commands I want to be able to give the object.
graph.addNode("A");
graph.addNode("B");
graph.addLink("A", "B");

</script>
</html>

新しいノードを追加するたびに、既存のすべてのノードに再度ラベルが付けられます。これらは互いに重なり合い、物事は醜くなり始めます。update()新しいノードを追加するときに関数function を呼び出すと、node.append(...)データセット全体に対してa を実行するので、これが理由を理解しています。追加するノードに対してのみこれを行う方法を理解することはできません...そして、明らかにnode.enter()単一の新しい要素を作成するためにしか使用できないため、ノードにバインドする必要がある追加の要素に対しては機能しません。どうすれば修正できますか?

この問題のいずれかに与えることができるガイダンスをありがとう!

以前に言及された他のいくつかのバグの原因をすばやく修正したため、編集されました

回答:


152

これを機能させることができない長い時間の後、私はようやく、どのドキュメントにもリンクされていないと思われるデモに出くわしました:http : //bl.ocks.org/1095795

ここに画像の説明を入力してください

このデモには、問題を解読するのに役立つキーが含まれていました。

に複数のオブジェクトを追加enter()するenter()には、変数にを割り当ててから追加します。意味あり。2番目の重要な部分は、ノードとリンクの配列がに基づいている必要があることです。force()そうでない場合、ノードが削除されて追加されると、グラフとモデルが同期しなくなります。

これは、代わりに新しい配列が作成されると、次の属性が不足するためです

  • index-ノード配列内のノードのゼロから始まるインデックス。
  • x-現在のノード位置のx座標。
  • y-現在のノード位置のy座標。
  • px-前のノード位置のx座標。
  • py-前のノード位置のy座標。
  • fixed-ノードの位置がロックされているかどうかを示すブール値。
  • weight-ノードの重み。関連するリンクの数。

これらの属性はへの呼び出しには厳密には必要ありませんforce.nodes()が、これらが存在しない場合、最初の呼び出しでによってランダムに初期化さforce.start()れます。

誰かが好奇心を持っている場合、作業コードは次のようになります。

<script type="text/javascript">

function myGraph(el) {

    // Add and remove elements on the graph object
    this.addNode = function (id) {
        nodes.push({"id":id});
        update();
    }

    this.removeNode = function (id) {
        var i = 0;
        var n = findNode(id);
        while (i < links.length) {
            if ((links[i]['source'] === n)||(links[i]['target'] == n)) links.splice(i,1);
            else i++;
        }
        var index = findNodeIndex(id);
        if(index !== undefined) {
            nodes.splice(index, 1);
            update();
        }
    }

    this.addLink = function (sourceId, targetId) {
        var sourceNode = findNode(sourceId);
        var targetNode = findNode(targetId);

        if((sourceNode !== undefined) && (targetNode !== undefined)) {
            links.push({"source": sourceNode, "target": targetNode});
            update();
        }
    }

    var findNode = function (id) {
        for (var i=0; i < nodes.length; i++) {
            if (nodes[i].id === id)
                return nodes[i]
        };
    }

    var findNodeIndex = function (id) {
        for (var i=0; i < nodes.length; i++) {
            if (nodes[i].id === id)
                return i
        };
    }

    // set up the D3 visualisation in the specified element
    var w = $(el).innerWidth(),
        h = $(el).innerHeight();

    var vis = this.vis = d3.select(el).append("svg:svg")
        .attr("width", w)
        .attr("height", h);

    var force = d3.layout.force()
        .gravity(.05)
        .distance(100)
        .charge(-100)
        .size([w, h]);

    var nodes = force.nodes(),
        links = force.links();

    var update = function () {

        var link = vis.selectAll("line.link")
            .data(links, function(d) { return d.source.id + "-" + d.target.id; });

        link.enter().insert("line")
            .attr("class", "link");

        link.exit().remove();

        var node = vis.selectAll("g.node")
            .data(nodes, function(d) { return d.id;});

        var nodeEnter = node.enter().append("g")
            .attr("class", "node")
            .call(force.drag);

        nodeEnter.append("image")
            .attr("class", "circle")
            .attr("xlink:href", "https://d3nwyuy0nl342s.cloudfront.net/images/icons/public.png")
            .attr("x", "-8px")
            .attr("y", "-8px")
            .attr("width", "16px")
            .attr("height", "16px");

        nodeEnter.append("text")
            .attr("class", "nodetext")
            .attr("dx", 12)
            .attr("dy", ".35em")
            .text(function(d) {return d.id});

        node.exit().remove();

        force.on("tick", function() {
          link.attr("x1", function(d) { return d.source.x; })
              .attr("y1", function(d) { return d.source.y; })
              .attr("x2", function(d) { return d.target.x; })
              .attr("y2", function(d) { return d.target.y; });

          node.attr("transform", function(d) { return "translate(" + d.x + "," + d.y + ")"; });
        });

        // Restart the force layout.
        force.start();
    }

    // Make it all go
    update();
}

graph = new myGraph("#graph");

// You can do this from the console as much as you like...
graph.addNode("Cause");
graph.addNode("Effect");
graph.addLink("Cause", "Effect");
graph.addNode("A");
graph.addNode("B");
graph.addLink("A", "B");

</script>

1
新しいデータが追加されるときのforce.start()代わりに使用することがforce.resume()鍵でした。どうもありがとう!
Mouagip 2014年

これは素晴らしいです。それはズームレベルを自動縮尺た場合、すべてが、それはで描いたボックスのサイズにフィットして(?多分、すべてのフィットゴマ電荷を減らす)クールに。
ロブ・グラント

1
クリーンなコード例の場合は+1。オブジェクトに動作をカプセル化する方法を示しているため、ボストック氏の例よりも気に入っています。よくやった。(D3例ライブラリに追加することを検討?)
fearless_fool

美しいです!私は数日間、d3でforceGraphを使用する方法を学習しています。これは、私が見た中で最も美しい方法です。本当にありがとうございました!
Lucas Azevedo
弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.