AngularJS承诺:在工厂中放置承诺,以便全局可访问且可编辑*

时间:2017-01-06 17:01:13

标签: angularjs promise factory

我正在尝试从外部JSON文件中提取数据并将其显示给用户查看。通过各种操作,用户将能够更改从JSON文件返回的数据,而无需将这些更改写入文件(在此示例中,通过单击div将值递增1)。我创建了一个承诺服务,成功地提取数据并显示它。我甚至可以得到它,以便可以在各个控制器中更改数据。

这就是我陷入困境的地方:我找不到对PromiseService中的数据进行任何更改的方法,因此更改无法全局传播。如何使控制器级别的promise数据中的任何更改都反映在PromiseService中,从而反映在应用程序中的任何数据绑定中?我是承诺的新手,所以我对一种完全不同的方法持开放态度。

Plunker

HTML:

<body ng-app="pageApp" ng-controller="pageCtrl" nd-model="items">
    {{items}}
    <div class="button" ng-controller="buttonCtrl" ng-click="incrementValues()">
        Click to increment:
        <br>{{items}}
    </div>
</body>

PromiseService:

pageApp.factory('PromiseService', function($http) { 
    var getPromise = function() {
        return $http.get('items.json').then(function(response) {
            return response.data;
        });
    };

    return {
        getPromise: getPromise
    };
});

按钮控制器(Plunker中的页面控制器):

pageApp.controller('buttonCtrl', function($scope, PromiseService) {
    $scope.incrementValues = function()
    {
        PromiseService.getPromise().then(function(data) {
            $scope.items = data;
            for(var i = 0; i < data.items.length; i++)
            {
                data.items[i]['value']++;
            }
        }).catch(function() {
        });
    };
});

incrementValues函数第一次成功运行,但每次连续点击都会重新拉出promise并重置数据。总结一下:如何在PromiseService中反映增量值,而不是局部变量?

2 个答案:

答案 0 :(得分:2)

您可以在工厂中添加private存储商品的属性。然后创建3种不同的方法来更新和访问该属性。

    pageApp.factory('PromiseService', function($http) { 
        var items = {}; // [] in case it is an array

        var updateData = function(updatedData){
            items = updatedData;
        }

        var getUpdateData = function(){
            return items;
        }

        var getPromise = function() {
            return $http.get('items.json').then(function(response) {
                items = response.data;
                return response.data;

            });
        };

        return {
            getPromise: getPromise,
            updateData : updateData,
            getUpdateData : getUpdateData 
        };
    });


pageApp.controller('buttonCtrl', function($scope, PromiseService) {

    $scope.items = [];

   //You should call this method to retrieve the data from the json file
   $scope.getData = function(){
        PromiseService.getPromise().then(function(data) {
            $scope.items = data;            
        }).catch(function() {
        });

   }

   $scope.incrementValues = function(){
       for(var i = 0; i < $scope.items.length; i++){
           $scope.items[i]['value']++;
       }     
       PromiseService.updateData($scope.items); //This could be skipped in case you do not want to 'store' these changes. 
   };
});

然后在其他控制器中,您可以使用相同的服务来检索updated Data,如下所示:

$scope.items = PromiService.PromiseService();

将来您还可以创建一个新方法来更新json本身而不是内部存储

答案 1 :(得分:0)

您的函数每次调用时都会创建一个新的$ http调用,因此每次调用时都会返回一个新的承诺,包含新数据。

每次都需要返回相同的承诺:

var thePromise = $http.get('items.json').then(function(response) {
    return response.data;
});

var getPromise = function() {
    return thePromise;
};