7

我想要點擊submit按鈕並在輸入字段中輸入新值後,我的網絡d3.js圖將基於新生成的新的輸入值。在這裏,在下面,你可以找到我的例子代碼:結合angularJS和d3.js:提交新的輸入參數後刷新一個圖

GenerateGraph.js此文件包含了一堆的基礎上,提交輸入值,其產生功能(randomGraph)。然後該圖需要在瀏覽器中刷新。

function degree(node,list){ 
    var deg=new Array(); 
    for (var i=0; i<node.length; i++){ 
    var count=0; 
    for (var j=0; j<list.length; j++){ 
     if (node[i]==list[j][0] || node[i]==list[j][1]){ 
      count++; 
     } 
    } 
    deg.push(count); 
    } 
    return deg; 
} 
function randomGraph (n, m) { //creates a random graph on n nodes and m links 
    var graph={}; 
    var nodes = d3.range(n).map(Object), 
     list = randomChoose(unorderedPairs(d3.range(n)), m), 
     links = list.map(function (a) { return {source: a[0], target: a[1]} }); 
    graph={ 
    Node:nodes, 
    ListEdges:list, 
    Links:links 
    } 
    return graph; 
} 

function randomChoose (s, k) { // returns a random k element subset of s 
    var a = [], i = -1, j; 
    while (++i < k) { 
    j = Math.floor(Math.random() * s.length); 
    a.push(s.splice(j, 1)[0]); 
    }; 
    return a; 
} 

function unorderedPairs (s) { // returns the list of all unordered pairs from s 
    var i = -1, a = [], j; 
    while (++i < s.length) { 
    j = i; 
    while (++j < s.length) a.push([s[i],s[j]]) 
    }; 
    return a; 
} 

network.html

!DOCTYPE html> 
<html> 
<head> 
    <meta charset="utf-8">  
    <meta name="viewport" content="width=device-width, initial-scale=1"> 
    <link rel="stylesheet" href="https://fonts.googleapis.com/css?family=Tangerine"> 
    <link rel="stylesheet" type="text/css" href="https://maxcdn.bootstrapcdn.com/bootstrap/3.3.7/css/bootstrap.min.css"> 
    <script src="https://ajax.googleapis.com/ajax/libs/angularjs/1.6.4/angular.min.js"></script> 
    <title>graph</title> 

    <script src='http://d3js.org/d3.v3.min.js'></script>  
    <link rel="stylesheet" type="text/css" href="style.css"> 

</head> 
<body ng-app="myApp"> 
    <script src="GenerateGraph.js" type="text/javascript"></script> 
    <script src="svgGraph.js" type="text/javascript"></script> 
    <h1 class="title">Simulating a network</h1> 
    <div id="outer" ng-controller="MainCtrl" class="col-md-6"> 
     <network-inputs inputs="networkInputs" submit="submit(inputs)"></network-inputs> 
    </div> 
    <!--test --> 
    <script type="text/javascript"> 

     //get the input parameters for plotting 
     angular.module("myApp", []) 

     .directive('networkInputs', function() { 

     return { 
       restrict: 'E',     
       scope: { 
        inputs: '<', 
        submit: '&' 
       }, 
       link : link,    
       template: 
       '<h3 >Initialise new parameters to generate a network </h3>'+ 
        '<form ng-submit="submit({inputs: inputs})" class="form-inline">'+ 
         '<div class="form-group">'+ 
         '<label>Number of nodes</label>'+ 
         '<input type="number" min="10" class="form-control" ng-model="inputs.N" ng-required="true">'+ 
         '</div>'+ 
         '<div class="form-group">'+ 
         '<label>Number of links</label>'+ 
          '<input type="number" min="0.1" class="form-control" ng-model="inputs.m" ng-required="true">'+ 
         '</div>'+ 
         '<button style="color:black; margin: 1rem 4rem;" type="submit">Generate</button>' + 
        '</form>'}; 
     }) 
     .factory("initialiseNetwork",function(){ 
      var data = { 
          N: 20, 
          m: 50, 

         }; 

      return { 
       networkInputs:data 
      };  

     }) 

     .controller("MainCtrl", ['$scope','initialiseNetwork' ,function($scope,initialiseNetwork) { 

       $scope.networkInputs={}; 
       $scope.mySVG=function(){ 
         var graph=randomGraph($scope.networkInputs.N, $scope.networkInputs.m); 

       }; 

       function init(){ 
        $scope.networkInputs=initialiseNetwork.networkInputs; 
        //Run the function which generates the graph and plot it 

       } 
       init(); 

       $scope.submit = function(inputs) { 

        var dataObject = { 
         N: inputs.N, 
         m: inputs.m 
        }; 
        //lets simply log them but you can plot or smth other 
        console.log($scope.networkInputs); 

       } 

     }]); 

    </script> 
</body> 

</html> 

svgGraph.js

function link(scope,element, attrs){ 
    //SVG size 
    var width = 1800, 
    height = 1100; 

    // We only need to specify the dimensions for this container. 

    var vis = d3.select(element[0]).append('svg') 
       .attr('width', width) 
       .attr('height', height); 
    var force = d3.layout.force() 
         .gravity(.05) 
         .distance(100) 
         .charge(-100) 
         .size([width, height]); 
     // Extract the nodes and links from the data. 
    scope.$watch('val',function(newVal,oldVal){ 
        vis.selectAll('*').remove(); 
        if (!newVal){ 
         return; 
        } 

     var Glinks = newVal.links; 
     var W=degree(newVal.nodes,newVal.list); 

     var Gnodes = []; 
     var obj=newVal.nodes; 
     Object.keys(obj).forEach(function(key) { 
      Gnodes.push({"name":key, "count":W[key]}); 
     }); 

     //Creates the graph data structure 
     force.nodes(Gnodes) 
      .links(Glinks) 
      .linkDistance(function(d) { 
       return(0.1*Glinks.length); 
      })//link length 
      .start(); 
     //Create all the line svgs but without locations yet 
     var link = vis.selectAll(".link") 
      .data(Glinks) 
      .enter().append("line") 
      .attr("class", "link") 
      .style("stroke-width","0.3px"); 

     //Do the same with the circles for the nodes - no 
     var node = vis.selectAll(".node") 
      .data(Gnodes) 
      .enter().append("g") 
      .attr("class", "node") 
      .call(force.drag); 
     node.append("circle") 
      .attr("r", function(d){ 
       return d.count*0.5; 
      }) 
      .style("opacity", .3) 
      .style("fill", "red"); 
     //add degree of node as text 
     node.append("text") 
      .attr("text-anchor", "middle") 
      .text(function(d) { return d.count }) 
      .attr("font-family",'Raleway',"Tangerine"); 
     //Now we are giving the SVGs co-ordinates - the force layout is generating the co-ordinates which this code is using to update the attributes of the SVG elements 
     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 + ")"; }); 
     });  
    }); 
} 

plunkerhere鏈路。

+0

請你熱度把你的問題變成[mcve]?完成所有代碼是更多的工作。讓我們很容易幫助你 – ochi

+0

@ochi我向你保證我簡化了它,但我也需要這些部分一起工作。 – Dalek

+0

你能把小提琴演奏出來嗎? – carlcheel

回答

5

1 - 在指令中,你看val,它在控制器中是$ scope.data,所以我想你需要它爲每個提交的表單?然後就數據分配到$ scope.data每提交:

$scope.submit = function(inputs) { 
    var dataObject = { 
     N: inputs.N, 
     m: inputs.m 
    }; 
    $scope.data = randomGraph(dataObject.N, dataObject.m); 
} 

2 - 然後,在sgvGraph.js,在scope.watch,您使用VAR newVal.nodes和newVal.list ANF newVal.link這都是未定義的,因爲您使用{Node:..,Links:...,ListEdges:...}來構建對象。

3 - 應該在表單中添加novalidate並手動管理錯誤,因爲我無法使用分鐘= 「0.1」 鉻

這是工作plunkr:http://embed.plnkr.co/PbynuNCPM4Jv4lPmK8eW/

+0

非常感謝!它完美的作品。只有一件東西缺失。如何使用初始值在瀏覽器中生成一個繪圖?之前當我有獨立的繪圖函數不在指令中時,我在控制器的'init()'函數中調用它,並且頁面正在用初始值進行渲染。 – Dalek

+0

現在,要生成圖形,只需將$ scope.data更改爲由randomGraph()創建的數據,因此添加$ scope.data = randomGraph($ scope.networkInputs.N,$ scope.networkInputs.m );在init()函數中會工作,我更新了plnkr – Fetrarij