Angularjs กำลังโหลดหน้าจอตามคำขอ ajax


117

เมื่อใช้ Angularjs ฉันต้องแสดงหน้าจอโหลด (สปินเนอร์ธรรมดา) จนกว่าคำขอ ajax จะเสร็จสมบูรณ์ โปรดแนะนำแนวคิดใด ๆ ด้วยข้อมูลโค้ด




แนวทางปฏิบัติที่ดีที่สุดคือการใช้$httpProvider.interceptorsเนื่องจากสามารถจัดการเมื่อคำขอ ajax เริ่มต้นและเมื่อสิ้นสุด นั่นคือเหตุผลที่$watchไม่จำเป็นต้องตรวจจับอีกต่อไปเมื่อการโทรของ ajax เริ่มต้นและสิ้นสุด
Frank Myat พฤหัสบดี

แล้วคุณจะชำระเงินด้วยangular-httpsของโรงงานของฉันได้อย่างไรมันช่วยให้คุณควบคุมรถตักและ UI การแช่แข็งได้ดีขึ้น
Siddharth

คำตอบ:


210

แทนที่จะตั้งค่าตัวแปรขอบเขตเพื่อระบุสถานะการโหลดข้อมูลจะดีกว่าหากมีคำสั่งทำทุกอย่างให้คุณ:

angular.module('directive.loading', [])

    .directive('loading',   ['$http' ,function ($http)
    {
        return {
            restrict: 'A',
            link: function (scope, elm, attrs)
            {
                scope.isLoading = function () {
                    return $http.pendingRequests.length > 0;
                };

                scope.$watch(scope.isLoading, function (v)
                {
                    if(v){
                        elm.show();
                    }else{
                        elm.hide();
                    }
                });
            }
        };

    }]);

ด้วยคำสั่งนี้สิ่งที่คุณต้องทำคือกำหนดให้องค์ประกอบภาพเคลื่อนไหวในการโหลดเป็นแอตทริบิวต์ "กำลังโหลด":

<div class="loading-spiner-holder" data-loading ><div class="loading-spiner"><img src="..." /></div></div>

คุณสามารถมีสปินเนอร์โหลดได้หลายตัวบนหน้า ที่ไหนและอย่างไรในการจัดวางสปินเนอร์เหล่านั้นขึ้นอยู่กับคุณและคำสั่งจะเปิด / ปิดให้คุณโดยอัตโนมัติ


ที่นี่คำสั่ง "loading" ถูกเรียกสองครั้งโดยมี $ location.path () เริ่มจากหน้าปัจจุบันและถัดไปด้วยหน้าปลายทาง เหตุผลควรเป็นอย่างไร
Sanjay D

ยอดเยี่ยม นอกจากนี้คุณสามารถใช้ jquery toggle บน watch: elm.toggle (v) ฉันมีการตรวจสอบเซสชัน $ timeout และสำหรับฉันฉันต้องแสดงตัวหมุนหลังจากผ่านไปครึ่งวินาทีเท่านั้น ฉันจะใช้ css พร้อมการเปลี่ยนเพื่อแสดงสปินเนอร์ช้าลง
Joao Polo

7
หากคุณได้รับข้อผิดพลาด "elm.show () ไม่ใช่ฟังก์ชัน" คุณต้องเพิ่ม jquery ก่อนที่จะโหลดเชิงมุม
morpheus05

วิธีที่คุณทำ scope.watch ไม่ได้ผลสำหรับฉัน ฉันต้องทำแบบที่ jzm ทำ (โดยใช้ '' รอบชื่อและละเว้นคำนำหน้าขอบเขต) ความคิดใด ๆ ทำไม?
KingOfHypocrites

3
จะเกิดอะไรขึ้นถ้าฉันต้องการ async โหลดหลาย ๆ พื้นที่?
Vadim Ferderer

56

นี่คือตัวอย่าง ใช้วิธี ng-show อย่างง่ายด้วยบูล

HTML

<div ng-show="loading" class="loading"><img src="...">LOADING...</div>
<div ng-repeat="car in cars">
  <li>{{car.name}}</li>
</div>
<button ng-click="clickMe()" class="btn btn-primary">CLICK ME</button>

ANGULARJS

  $scope.clickMe = function() {
    $scope.loading = true;
    $http.get('test.json')
      .success(function(data) {
        $scope.cars = data[0].cars;
        $scope.loading = false;
    });
  }

แน่นอนว่าคุณสามารถย้ายโค้ด html ของกล่องโหลดไปไว้ในคำสั่งจากนั้นใช้ $ watch กับ $ scope.loading ในกรณีนี้:

HTML :

<loading></loading>

AngULARJS ไดเรคทีฟ :

  .directive('loading', function () {
      return {
        restrict: 'E',
        replace:true,
        template: '<div class="loading"><img src="..."/>LOADING...</div>',
        link: function (scope, element, attr) {
              scope.$watch('loading', function (val) {
                  if (val)
                      $(element).show();
                  else
                      $(element).hide();
              });
        }
      }
  })

PLUNK : http://plnkr.co/edit/AI1z21?p=preview


วิธีที่คุณทำ scope.watch ใช้ได้ผลสำหรับฉันในขณะที่แนวทางคำตอบที่ยอมรับไม่ได้
KingOfHypocrites

@jzm ช่างเป็นทางออกที่ยอดเยี่ยม😊น่าเสียดายที่โซลูชันนี้ใช้ui-routerไม่ได้ไม่รู้ทำไม
โม

ฉันทำสิ่งนี้ในโครงการหนึ่งของฉันได้ผลดีในอีกโครงการหนึ่งฉันคิดว่าเนื่องจากคำขอ ajax จำนวนมากloading=trueไม่ได้รับการตั้งค่า
alamnaryab

21

ฉันใช้ngProgressสำหรับสิ่งนี้

เพิ่ม 'ngProgress' ในการอ้างอิงของคุณเมื่อคุณรวมไฟล์สคริปต์ / css ไว้ใน HTML แล้ว เมื่อคุณทำเสร็จแล้วคุณสามารถตั้งค่าสิ่งนี้ได้ซึ่งจะทริกเกอร์เมื่อตรวจพบการเปลี่ยนแปลงเส้นทาง

angular.module('app').run(function($rootScope, ngProgress) {
  $rootScope.$on('$routeChangeStart', function(ev,data) {
    ngProgress.start();
  });
  $rootScope.$on('$routeChangeSuccess', function(ev,data) {
    ngProgress.complete();
  });
});

สำหรับคำขอ AJAX คุณสามารถทำสิ่งนี้ได้:

$scope.getLatest = function () {
    ngProgress.start();

    $http.get('/latest-goodies')
         .success(function(data,status) {
             $scope.latest = data;
             ngProgress.complete();
         })
         .error(function(data,status) {
             ngProgress.complete();
         });
};

อย่าลืมเพิ่ม 'ngProgress' ในการอ้างอิงตัวควบคุมก่อนที่จะทำเช่นนั้น และหากคุณกำลังทำคำขอ AJAX หลายรายการให้ใช้ตัวแปรที่เพิ่มขึ้นในขอบเขตแอปหลักเพื่อติดตามเมื่อคำขอ AJAX ของคุณเสร็จสิ้นก่อนที่จะเรียก 'ngProgress.complete ();'


15

การใช้ pendingRequests ไม่ถูกต้องเนื่องจากตามที่กล่าวไว้ในเอกสาร Angular คุณสมบัตินี้มีไว้เพื่อใช้เพื่อวัตถุประสงค์ในการดีบักเป็นหลัก

สิ่งที่ฉันแนะนำคือให้ใช้เครื่องดักฟังเพื่อดูว่ามีการเรียก Async ที่ใช้งานอยู่หรือไม่

module.config(['$httpProvider', function ($httpProvider) {
    $httpProvider.interceptors.push(function ($q, $rootScope) {
        if ($rootScope.activeCalls == undefined) {
            $rootScope.activeCalls = 0;
        }

        return {
            request: function (config) {
                $rootScope.activeCalls += 1;
                return config;
            },
            requestError: function (rejection) {
                $rootScope.activeCalls -= 1;
                return rejection;
            },
            response: function (response) {
                $rootScope.activeCalls -= 1;
                return response;
            },
            responseError: function (rejection) {
                $rootScope.activeCalls -= 1;
                return rejection;
            }
        };
    });
}]);

จากนั้นตรวจสอบว่า activeCalls เป็นศูนย์หรือไม่ในคำสั่งผ่าน $ watch

module.directive('loadingSpinner', function ($http) {
    return {
        restrict: 'A',
        replace: true,
        template: '<div class="loader unixloader" data-initialize="loader" data-delay="500"></div>',
        link: function (scope, element, attrs) {

            scope.$watch('activeCalls', function (newVal, oldVal) {
                if (newVal == 0) {
                    $(element).hide();
                }
                else {
                    $(element).show();
                }
            });
        }
    };
});

7

วิธีที่ดีที่สุดที่จะทำนี้คือการใช้interceptors การตอบสนองพร้อมกับกำหนดเองสั่ง และกระบวนการที่สามารถเพิ่มเติมปรับปรุงการใช้ผับ / ย่อยกลไกการใช้$ rootScope. $ ออกอากาศและ$ rootScope. $ ในวิธีการ

เนื่องจากกระบวนการทั้งหมดได้รับการบันทึกไว้ในบทความบล็อกที่เขียนดีฉันจะไม่ทำซ้ำที่นี่อีก โปรดดูบทความนั้นเพื่อจัดเตรียมการใช้งานที่คุณต้องการ


4

ในการอ้างอิงคำตอบนี้

https://stackoverflow.com/a/17144634/4146239

สำหรับฉันเป็นทางออกที่ดีที่สุด แต่มีวิธีหลีกเลี่ยงการใช้ jQuery

.directive('loading', function () {
      return {
        restrict: 'E',
        replace:true,
        template: '<div class="loading"><img src="http://www.nasa.gov/multimedia/videogallery/ajax-loader.gif" width="20" height="20" />LOADING...</div>',
        link: function (scope, element, attr) {
              scope.$watch('loading', function (val) {
                  if (val)
                      scope.loadingStatus = 'true';
                  else
                      scope.loadingStatus = 'false';
              });
        }
      }
  })

  .controller('myController', function($scope, $http) {
      $scope.cars = [];
      
      $scope.clickMe = function() {
        scope.loadingStatus = 'true'
        $http.get('test.json')
          .success(function(data) {
            $scope.cars = data[0].cars;
            $scope.loadingStatus = 'false';
        });
      }
      
  });
<body ng-app="myApp" ng-controller="myController" ng-init="loadingStatus='true'">
        <loading ng-show="loadingStatus" ></loading>
  
        <div ng-repeat="car in cars">
          <li>{{car.name}}</li>
        </div>
        <button ng-click="clickMe()" class="btn btn-primary">CLICK ME</button>
  
</body>

คุณต้องแทนที่ $ (element) .show (); และ (องค์ประกอบ) .show (); ด้วย $ scope.loadingStatus = 'true'; และ $ scope.loadingStatus = 'false';

คุณต้องใช้ตัวแปรนี้เพื่อตั้งค่าคุณสมบัติ ng-show ขององค์ประกอบ


4

typescript และ Angular Implementation

คำสั่ง

((): void=> {
    "use strict";
    angular.module("app").directive("busyindicator", busyIndicator);
    function busyIndicator($http:ng.IHttpService): ng.IDirective {
        var directive = <ng.IDirective>{
            restrict: "A",
            link(scope: Scope.IBusyIndicatorScope) {
                scope.anyRequestInProgress = () => ($http.pendingRequests.length > 0);
                scope.$watch(scope.anyRequestInProgress, x => {            
                    if (x) {
                        scope.canShow = true;
                    } else {
                        scope.canShow = false;
                    }
                });
            }
        };
        return directive;
    }
})();

ขอบเขต

   module App.Scope {
        export interface IBusyIndicatorScope extends angular.IScope {
            anyRequestInProgress: any;
            canShow: boolean;
        }
    }  

แบบ

<div id="activityspinner" ng-show="canShow" class="show" data-busyindicator>
</div>

CSS
#activityspinner
{
    display : none;
}
#activityspinner.show {
    display : block;
    position : fixed;
    z-index: 100;
    background-image : url('') 
    -ms-opacity : 0.4;
    opacity : 0.4;
    background-repeat : no-repeat;
    background-position : center;
    left : 0;
    bottom : 0;
    right : 0;
    top : 0;
}

3

หากคุณใช้ Restangular (ซึ่งยอดเยี่ยมมาก) คุณสามารถสร้างแอนิเมชั่นระหว่างการเรียก API ได้ นี่คือทางออกของฉัน เพิ่มตัวสกัดกั้นการตอบสนองและตัวสกัดการร้องขอที่ส่งการออกอากาศรูทสโคป จากนั้นสร้างคำสั่งเพื่อรับฟังคำตอบและคำขอนั้น:

         angular.module('mean.system')
  .factory('myRestangular',['Restangular','$rootScope', function(Restangular,$rootScope) {
    return Restangular.withConfig(function(RestangularConfigurer) {
      RestangularConfigurer.setBaseUrl('http://localhost:3000/api');
      RestangularConfigurer.addResponseInterceptor(function(data, operation, what, url, response, deferred) {
        var extractedData;
        // .. to look for getList operations
        if (operation === 'getList') {
          // .. and handle the data and meta data
          extractedData = data.data;
          extractedData.meta = data.meta;
        } else {
          extractedData = data.data;
        }
        $rootScope.$broadcast('apiResponse');
        return extractedData;
      });
      RestangularConfigurer.setRequestInterceptor(function (elem, operation) {
        if (operation === 'remove') {
          return null;
        }
        return (elem && angular.isObject(elem.data)) ? elem : {data: elem};
      });
      RestangularConfigurer.setRestangularFields({
        id: '_id'
      });
      RestangularConfigurer.addRequestInterceptor(function(element, operation, what, url) {
        $rootScope.$broadcast('apiRequest');
        return element;
      });
    });
  }]);

นี่คือคำสั่ง:

        angular.module('mean.system')
  .directive('smartLoadingIndicator', function($rootScope) {
    return {
      restrict: 'AE',
      template: '<div ng-show="isAPICalling"><p><i class="fa fa-gear fa-4x fa-spin"></i>&nbsp;Loading</p></div>',
      replace: true,
      link: function(scope, elem, attrs) {
        scope.isAPICalling = false;

        $rootScope.$on('apiRequest', function() {
          scope.isAPICalling = true;
        });
        $rootScope.$on('apiResponse', function() {
          scope.isAPICalling = false;
        });
      }
    };
  })
;

การตอบสนองของ API จะฆ่าภาพเคลื่อนไหว คุณควรจัดเก็บข้อมูลเพิ่มเติมเกี่ยวกับคำขอ - การตอบกลับและตอบสนองเฉพาะสำหรับการตอบกลับที่เริ่มต้นโดยการร้องขอ
Krzysztof Safjanowski

3

รวมสิ่งนี้ไว้ใน "app.config" ของคุณ:

 $httpProvider.interceptors.push('myHttpInterceptor');

และเพิ่มรหัสนี้:

app.factory('myHttpInterceptor', function ($q, $window,$rootScope) {
    $rootScope.ActiveAjaxConectionsWithouthNotifications = 0;
    var checker = function(parameters,status){
            //YOU CAN USE parameters.url TO IGNORE SOME URL
            if(status == "request"){
                $rootScope.ActiveAjaxConectionsWithouthNotifications+=1;
                $('#loading_view').show();
            }
            if(status == "response"){
                $rootScope.ActiveAjaxConectionsWithouthNotifications-=1;

            }
            if($rootScope.ActiveAjaxConectionsWithouthNotifications<=0){
                $rootScope.ActiveAjaxConectionsWithouthNotifications=0;
                $('#loading_view').hide();

            }


    };
return {
    'request': function(config) {
        checker(config,"request");
        return config;
    },
   'requestError': function(rejection) {
       checker(rejection.config,"request");
      return $q.reject(rejection);
    },
    'response': function(response) {
         checker(response.config,"response");
      return response;
    },
   'responseError': function(rejection) {
        checker(rejection.config,"response");
      return $q.reject(rejection);
    }
  };
});

2

ใช้เชิงมุมไม่ว่าง :

เพิ่ม cgBusy ในแอป / โมดูลของคุณ:

angular.module('your_app', ['cgBusy']);

เพิ่มคำสัญญาของคุณกับscope:

function MyCtrl($http, User) {
  //using $http
  this.isBusy = $http.get('...');
  //if you have a User class based on $resource
  this.isBusy = User.$save();
}

ในเทมเพลต html ของคุณ:

<div cg-busy="$ctrl.isBusy"></div>

2

นอกจากนี้ยังมีการสาธิตที่ดีที่แสดงให้เห็นว่าคุณสามารถใช้Angularjsภาพเคลื่อนไหวในโครงการของคุณได้อย่างไร

ลิงค์อยู่ที่นี่ (ดูที่มุมบนซ้าย)

มันเป็นโอเพ่นซอร์ส นี่คือลิงค์สำหรับดาวน์โหลด

และนี่คือการเชื่อมโยงสำหรับการกวดวิชา ;

ประเด็นของฉันคือไปข้างหน้าและดาวน์โหลดไฟล์ต้นฉบับจากนั้นดูว่าพวกเขาใช้สปินเนอร์อย่างไร พวกเขาอาจใช้ aproach ที่ดีกว่าเล็กน้อย ดังนั้นชำระเงินโครงการนี้


1

ตัวอย่างง่ายๆที่นี่ฉันตั้งค่าเมาส์ไว้รอเมื่อ ajax เริ่มและตั้งค่าเป็นอัตโนมัติเมื่อ ajax สิ้นสุดลง

$httpProvider.interceptors.push(function($document) {
return {
 'request': function(config) {
     // here ajax start
     // here we can for example add some class or show somethin
     $document.find("body").css("cursor","wait");

     return config;
  },

  'response': function(response) {
     // here ajax ends
     //here we should remove classes added on request start

     $document.find("body").css("cursor","auto");

     return response;
  }
};
});

app.configรหัสจะต้องมีการเพิ่มในการปรับแต่งที่โปรแกรม ฉันแสดงวิธีเปลี่ยนเมาส์ในสถานะการโหลด แต่ในนั้นเป็นไปได้ที่จะแสดง / ซ่อนเนื้อหาตัวโหลดใด ๆ หรือเพิ่มลบคลาส css บางตัวที่แสดงตัวโหลด

Interceptor จะทำงานทุกครั้งที่เรียก ajax ดังนั้นไม่จำเป็นต้องสร้างตัวแปรบูลีนพิเศษ ($ scope.loading = true / false เป็นต้น) ในทุกๆการเรียก http

Interceptor ใช้สร้างใน angular jqLite https://docs.angularjs.org/api/ng/function/angular.elementจึงไม่จำเป็นต้องใช้ Jquery


1

สร้างคำสั่งด้วยแอตทริบิวต์การแสดงและขนาด (คุณสามารถเพิ่มได้อีกด้วย)

    app.directive('loader',function(){
    return {
    restrict:'EA',
    scope:{
        show : '@',
      size : '@'
    },
    template : '<div class="loader-container"><div class="loader" ng-if="show" ng-class="size"></div></div>'
  }
})

และใน html ใช้เป็น

 <loader show="{{loader1}}" size="sm"></loader>

ในตัวแปรshowจะส่งผ่านจริงเมื่อคำสัญญาใด ๆ กำลังทำงานและทำให้เป็นเท็จเมื่อคำขอเสร็จสมบูรณ์ Active demo - ตัวอย่างคำสั่ง Angular Loader ตัวอย่างการสาธิตใน JsFiddle


0

ฉันสร้างคำตอบของ @ DavidLin เล็กน้อยเพื่อทำให้ง่ายขึ้น - ลบการพึ่งพา jQueryในคำสั่ง ฉันสามารถยืนยันได้ว่าใช้งานได้เมื่อใช้ในแอปพลิเคชันการผลิต

function AjaxLoadingOverlay($http) {

    return {
        restrict: 'A',
        link: function ($scope, $element, $attributes) {

            $scope.loadingOverlay = false;

            $scope.isLoading = function () {
                return $http.pendingRequests.length > 0;
            };

            $scope.$watch($scope.isLoading, function (isLoading) {
                $scope.loadingOverlay = isLoading;
            });
        }
    };
}   

ฉันใช้ng-showแทนการเรียก jQuery เพื่อซ่อน / แสดงไฟล์<div>.

นี่คือสิ่ง<div>ที่ฉันวางไว้ใต้<body>แท็กเปิด:

<div ajax-loading-overlay class="loading-overlay" ng-show="loadingOverlay">
    <img src="Resources/Images/LoadingAnimation.gif" />
</div>

และนี่คือ CSS ที่ให้การซ้อนทับเพื่อบล็อก UI ในขณะที่กำลังทำการเรียก $ http:

.loading-overlay {
    position: fixed;
    z-index: 999;
    height: 2em;
    width: 2em;
    overflow: show;
    margin: auto;
    top: 0;
    left: 0;
    bottom: 0;
    right: 0;
}

.loading-overlay:before {
    content: '';
    display: block;
    position: fixed;
    top: 0;
    left: 0;
    width: 100%;
    height: 100%;
    background-color: rgba(0,0,0,0.3);
}

/* :not(:required) hides these rules from IE9 and below */
.loading-overlay:not(:required) {
    font: 0/0 a;
    color: transparent;
    text-shadow: none;
    background-color: transparent;
    border: 0;
}

เครดิต CSS ไปที่ @Steve Seeger's - โพสต์ของเขา: https://stackoverflow.com/a/35470281/335545


0

คุณสามารถเพิ่มเงื่อนไขแล้วเปลี่ยนผ่านรูทสโคป ก่อนที่คุณจะร้องขอ ajax คุณเพียงแค่เรียก $ rootScope $ emit ('stopLoader');

angular.module('directive.loading', [])
        .directive('loading',   ['$http', '$rootScope',function ($http, $rootScope)
        {
            return {
                restrict: 'A',
                link: function (scope, elm, attrs)
                {
                    scope.isNoLoadingForced = false;
                    scope.isLoading = function () {
                        return $http.pendingRequests.length > 0 && scope.isNoLoadingForced;
                    };

                    $rootScope.$on('stopLoader', function(){
                        scope.isNoLoadingForced = true;
                    })

                    scope.$watch(scope.isLoading, function (v)
                    {
                        if(v){
                            elm.show();
                        }else{
                            elm.hide();
                        }
                    });
                }
            };

        }]);

นี่ไม่ใช่วิธีแก้ปัญหาที่ดีที่สุดอย่างแน่นอน แต่ก็ยังใช้งานได้

โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.