AngularJS数据绑定 - 监视周期不触发

时间:2015-10-16 01:14:47

标签: javascript angularjs d3.js nvd3.js

假设以下JSFiddle:https://jsfiddle.net/pmgq00fm/1/

我希望我的NVD3图表能够实时更新,基于第39行的setInterval()更新指令绑定的数据。 以下是有关架构和代码的快速指示:

  • 该指令属于ThisController
  • 的范围
  • 使用指令
  • 中的'='双向绑定数据
  • 只要指令在$ watch循环期间检测到数据发生变化,就应该在图表上调用更新函数。
  • $ watch是一块很深的手表,因此应该检测到值的变化
  • 在指令中设置间隔,以在控制台中打印值的更改。每当调用updateFnc时,控制器都会打印更新的数组。
  • 指令从不中的数据中的值与控制器中的数据匹配,因为我忽略了某些原因。
  • 在控制器和指令中修改这些值。
  • 指令中的$ watch仅在执行时调用
  • 代码基于this tutorial,并尊重其中解释的数据绑定所需的所有步骤。

要进行调试,请运行JSFiddle并打开控制台以查看对象和调试打印。

HTML:

<div id="stats-container" ng-app="myApp" ng-controller="ThisController">
    <div id="nvd3-test" nvd3-discrete-bar data="data.graph">       
</div> 

JS:

myControllers.controller('ThisController', ['$scope', function ThisController($scope){
      $scope.data = { graph : [ ... ] };

      updateFnc = function(data){
        for(var i = 0; i < data[0].values.length ; i++){
          data[0].values[i].value = Math.random();
        }
        console.log(Date.now()/1000 + ": In Controller");
        console.log(data);
      };

      setInterval(function(){
        updateFnc($scope.data.graph);
      }, 1000);
 }]);

myServices.factory('NVD3Wrapper', ['NVD3S', 'D3S', '$q', '$rootScope', function NVD3Wrapper(nvd3s, d3s, $q, $rootScope){
  return {
    nvd3: nvd3s,
    d3: d3s,
    discreteBarChart : function(data, config){
        var $nvd3 = this.nvd3,
            $d3 = this.d3,
            $nvd3w = this, //In order to resolve the nvd3w in other scopes.
            d = $q.defer(); //Creating a promise because chart rendering is asynchronous
        $nvd3.addGraph(function() {
          ...
        });
        return {
          chart: function() { return d.promise; } //returns the chart once rendered.
        };
    },
    _onRenderEnd: function(d, chart){
      $rootScope.$apply(function() { d.resolve(chart); });
    },
  };
}]);

myDirectives.directive('nvd3DiscreteBar', ['NVD3Wrapper', function(nvd3w){
  return {
    restrict: 'EA',
    scope: {
      data: '=' // bi-directional data-binding
    },
    link: function(scope, element, attrs) {
      var chart,
          config = {
            target: element,
          };
      var wrapper = nvd3w.discreteBarChart(scope.data, config);
      wrapper.chart().then(function(chart){
        scope.$watch(function() { return scope.data; }, function(newValue, oldValue) {
          console.log(Date.now()/1000 + ": In Directive $watch");
          if (newValue)
              chart.update();
        }, true);
      });
      //For testing
      setInterval(function(){ console.log(Date.now()/1000 + ": In Directive"); console.log(scope.data); }, 1000);
    }
  };
}]);

任何帮助将不胜感激!非常感谢!

编辑:新的JSFiddle与Andrew Shirley的回答:https://jsfiddle.net/pmgq00fm/3/

1 个答案:

答案 0 :(得分:2)

添加行

$scope.$apply()

更新功能。这是因为当在javascript中更新变量时,在常见角度函数之外,角度不知道变化,因此不会浪费任何刷新DOM的努力。这会强制消化周期,这应该刷新你所看到的内容。

编辑:真正了解你为什么实际使用scope.apply非常重要,而且我不觉得我已经非常好地描述了它。这是一篇比我更好的文章。

http://jimhoskins.com/2012/12/17/angularjs-and-apply.html

我指出这一点是因为你需要注意,如果你在一个与角度密切相关的函数内(例如通过ng-click调用的东西),那么如果你试图使用一个范围。$ apply你会得到javascript错误,因为你已经处于摘要周期的中间。