使用 Angularjs ,我需要显示一个加载屏幕(一个简单的微调器),直到 ajax 请求完成。请用代码片段提出任何想法。
15 回答
与其设置范围变量来指示数据加载状态,不如让指令为您完成所有事情:
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>
您可以在页面上有多个加载微调器。在哪里以及如何布置这些微调器取决于您,指令会自动为您打开/关闭它。
这是一个例子。它使用带有布尔值的简单 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 代码移动到一个指令中,然后在 $scope.loading 上使用 $watch。在这种情况下:
HTML:
<loading></loading>
角度指令:
.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();
});
}
}
})
我为此使用ngProgress。
在 HTML 中包含脚本/css 文件后,将“ngProgress”添加到您的依赖项中。一旦你这样做了,你可以设置这样的东西,当检测到路线变化时会触发。
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();”。
使用 pendingRequests 是不正确的,因为如 Angular 文档中所述,此属性主要用于调试目的。
我建议使用拦截器来了解是否有任何活动的异步调用。
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;
}
};
});
}]);
然后通过 $watch 检查指令中的 activeCalls 是否为零。
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();
}
});
}
};
});
最好的方法是使用响应拦截器和自定义指令。并且可以使用$rootScope.$broadcast和$rootScope.$on方法使用pub/sub机制进一步改进该过程。
由于整个过程都记录在一篇写得很好的博客文章中,我将不再在这里重复。请参阅该文章以提出您需要的实现。
参考这个答案
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 属性。
打字稿和角度实现
指示
((): 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;
}
如果您使用的是 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> 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;
});
}
};
})
;
将此包含在您的“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);
}
};
});
使用angular-busy:
将 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>
这里是简单的拦截器示例,我在 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 类。
拦截器将在每个 ajax 调用上运行,因此无需在每个 http 调用上创建特殊的布尔变量( $scope.loading=true/false 等)。
拦截器使用内置的 angular jqLite https://docs.angularjs.org/api/ng/function/angular.element,因此不需要 Jquery。
使用 show 和 size 属性创建指令(您也可以添加更多)
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>
在任何 Promise 运行时在show变量中传递true ,并在请求完成时将其设为 false 。活动演示 - JsFiddle 中的 Angular Loader 指令示例演示
我在@DavidLin's answer的基础上做了一点简化——删除了指令中对 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;
});
}
};
}
我使用 ang-show
而不是 jQuery 调用来隐藏/显示<div>
.
这是我放在开始标签<div>
下方的内容:<body>
<div ajax-loading-overlay class="loading-overlay" ng-show="loadingOverlay">
<img src="Resources/Images/LoadingAnimation.gif" />
</div>
这是在进行 $http 调用时提供覆盖以阻止 UI 的 CSS:
.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
您可以添加一个条件,然后通过 rootscope 更改它。在您的 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();
}
});
}
};
}]);
这绝对不是最好的解决方案,但它仍然有效。