Используя Angularjs, мне нужно показать экран загрузки (простой счетчик) до завершения запроса ajax. Пожалуйста, предложите любую идею с фрагментом кода.
Экран загрузки Angularjs по запросу ajax
Ответ 1
Вместо того, чтобы настраивать переменную области видимости для указания статуса загрузки данных, лучше иметь директиву для вас все:
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>
На странице может быть несколько роликов загрузки. где и как правильно расположить эти прядильщики, и директива просто автоматически включит/выключит вас.
Ответ 2
Вот пример. Он использует простой метод ng-show с bool.
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();
});
}
}
})
Ответ 3
Я использую ngProgress для этого.
Добавьте 'ngProgress' к вашим зависимостям, как только вы включили файлы script/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();".
Ответ 4
использование 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();
}
});
}
};
});
Ответ 5
Лучший способ сделать это - использовать ответные перехватчики вместе с пользовательской директивой . И этот процесс можно дополнительно улучшить с помощью механизма pub/sub с использованием $rootScope. $Broadcast и $rootScope. $On methods.
Как весь процесс документирован в хорошо написанной статье в блоге, я не собираюсь повторять это снова. Пожалуйста, обратитесь к этой статье, чтобы придумать необходимую вам реализацию.
Ответ 6
Ссылаясь на этот ответ
fooobar.com/questions/55955/...
Для меня это лучшее решение, но есть способ избежать использования 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>
Ответ 7
Typescript и Angular Реализация
директива
((): 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;
}
Ответ 8
Также есть хорошая демонстрация, которая показывает, как вы можете использовать анимацию Angularjs
в вашем проекте.
Ссылка здесь (см. Верхний левый угол).
Это открытый источник. Вот ссылка на скачивание
А вот ссылка на учебник;
Моя точка зрения заключается в том, чтобы скачать исходные файлы и посмотреть, как они реализовали счетчик. Они могли бы использовать немного лучший подход. Итак, зацените этот проект.
Ответ 9
Если вы используете 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;
});
}
};
})
;
Ответ 10
Включите это в свой "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);
}
};
});
Ответ 11
Используйте 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>
Ответ 12
Здесь простой пример перехватчика, я нажимаю кнопку мыши на ожидание, когда запускается 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-вызова.
Перехватчик использует построенный в angular jqLite https://docs.angularjs.org/api/ng/function/angular.element, поэтому нет необходимости в JQuery.
Ответ 13
Создайте директиву с атрибутами 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>
В переменной show передается true, когда выполняется какое-либо обещание и делает это ложным, когда запрос завершен. Активная демонстрация - Angular Демоверсия примера с загрузчиком в JsFiddle
Ответ 14
Я построил на @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, который обеспечивает наложение для блокировки пользовательского интерфейса при вызове $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 - его сообщение: fooobar.com/questions/55956/...
Ответ 15
Вы можете добавить условие, а затем изменить его с помощью корнеплодов. Перед вашим запросом 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();
}
});
}
};
}]);
Это, безусловно, не лучшее решение, но оно все равно будет работать.