Im new для Angular2, и мне было интересно, есть ли способ показать индикатор активности для каждого HTTP-запроса и скрыть представления до завершения?
Angular2: показать индикатор активности для каждого HTTP-запроса и скрыть представления до завершения
Ответ 1
Один из способов - написать перехватчик для Angular2 Http. Создав свой собственный экземпляр http, вы можете поменять его, когда вы загружаете свое приложение с помощью метода "предоставить". Как только это будет сделано, можно создать сервис PubSub для публикации и подписания этих событий с вашего перехватчика Http и испускать до и после событий при каждом запросе.
Ожидаемый пример можно увидеть на Plunker
Перехватчик:
import {Injectable} from 'angular2/core';
import {HTTP_PROVIDERS, Http, Request, RequestOptionsArgs, Response, XHRBackend, RequestOptions, ConnectionBackend, Headers} from 'angular2/http';
import 'rxjs/Rx';
import {PubSubService} from './pubsubService';
@Injectable()
export class CustomHttp extends Http {
_pubsub: PubSubService
constructor(backend: ConnectionBackend, defaultOptions: RequestOptions, pubsub: PubSubService) {
super(backend, defaultOptions);
this._pubsub = pubsub;
}
request(url: string | Request, options?: RequestOptionsArgs): Observable<Response> {
return this.intercept(super.request(url, options));
}
get(url: string, options?: RequestOptionsArgs): Observable<Response> {
return this.intercept(super.get(url,options));
}
post(url: string, body: string, options?: RequestOptionsArgs): Observable<Response> {
return this.intercept(super.post(url, body, this.getRequestOptionArgs(options)));
}
put(url: string, body: string, options?: RequestOptionsArgs): Observable<Response> {
return this.intercept(super.put(url, body, this.getRequestOptionArgs(options)));
}
delete(url: string, options?: RequestOptionsArgs): Observable<Response> {
return this.intercept(super.delete(url, options));
}
getRequestOptionArgs(options?: RequestOptionsArgs) : RequestOptionsArgs {
if (options == null) {
options = new RequestOptions();
}
if (options.headers == null) {
options.headers = new Headers();
}
options.headers.append('Content-Type', 'application/json');
return options;
}
intercept(observable: Observable<Response>): Observable<Response> {
this._pubsub.beforeRequest.emit("beforeRequestEvent");
//this will force the call to be made immediately..
observable.subscribe(
null,
null,
() => this._pubsub.afterRequest.emit("afterRequestEvent");
);
return observable
}
}
Излучатели
import {Subject } from 'rxjs/Subject';
export class RequestEventEmitter extends Subject<String>{
constructor() {
super();
}
emit(value) { super.next(value); }
}
export class ResponseEventEmitter extends Subject<String>{
constructor() {
super();
}
emit(value) { super.next(value); }
}
PubSubService
import {Injectable} from 'angular2/core';
import {RequestEventEmitter, ResponseEventEmitter} from './emitter';
@Injectable()
export class PubSubService{
beforeRequest:RequestEventEmitter;
afterRequest:ResponseEventEmitter;
constructor(){
this.beforeRequest = new RequestEventEmitter();
this.afterRequest = new ResponseEventEmitter();
}
}
Загрузочное приложение
//main entry point
import {bootstrap} from 'angular2/platform/browser';
import {provide} from 'angular2/core';
import {Http, HTTP_PROVIDERS, XHRBackend, RequestOptions} from 'angular2/http';
import {HelloWorldComponent} from './hello_world';
import {CustomHttp} from './customhttp';
import {PubSubService} from './pubsubService'
bootstrap(HelloWorldComponent, [HTTP_PROVIDERS,PubSubService,
provide(Http, {
useFactory: (backend: XHRBackend, defaultOptions: RequestOptions, pubsub: PubSubService)
=> new CustomHttp(backend, defaultOptions, pubsub),
deps: [XHRBackend, RequestOptions, PubSubService]
})
]).catch(err => console.error(err));
Теперь в вашем компоненте загрузки так же просто, как подписка на события и установка свойства для показа или нет
export class LoaderComponent implements OnInit {
showLoader = false;
_pubsub:PubSubService;
constructor(pubsub: PubSubService) {
this._pubsub = pubsub;
}
ngOnInit() {
this._pubsub.beforeRequest.subscribe(data => this.showLoader = true);
this._pubsub.afterRequest.subscribe(data => this.showLoader = false);
}
}
В то время как это заканчивается тем, что это немного больше кода, если вы хотите, чтобы вас уведомляли по каждому запросу в вашем приложении, это сделало бы это. Одна вещь, которую следует отметить с помощью перехватчика, заключается в том, что подписка выполняется для каждого запроса немедленно, все запросы будут выполнены, что может быть не так, как вам нужно в конкретных случаях. Решением этого является поддержка обычного Angular2 Http и использование CustomHttp в качестве второго варианта, который можно было бы впрыснуть там, где это необходимо. Я думаю, что в большинстве случаев немедленная подписка будет работать нормально. Я хотел бы услышать примеры того, когда это не будет.
Ответ 2
Да, вам нужно обработать это для каждого вида:
- У вас может быть служба для HTTP-запроса, которая вернет Наблюдаемый
- В компоненте у вас будет состояние загрузки
- Вам необходимо установить для состояния загрузки значение true, прежде чем запрашивать данные из сервера, затем установите значение false, когда выборка данных выполнена.
- В шаблоне используйте ngIf, чтобы скрыть/показать загрузку или содержимое
Пример:
Сервис:
@Injectable()
export class DataService {
constructor(private http: Http) { }
getData() {
return this.http.get('http://jsonplaceholder.typicode.com/posts/2');
}
}
Компонент:
@Component({
selector: 'my-app',
template : `
<div *ngIf="loading == true" class="loader">Loading..</div>
<div *ngIf="loading == false">Content.... a lot of content <br> more content</div>`
})
export class App {
loading: boolean;
constructor(private dataService: DataService) { }
ngOnInit() {
// Start loading Data from the Server
this.loading = true;
this.dataService.getData().delay(1500).subscribe(
requestData => {
// Data loading is Done
this.loading = false;
console.log('AppComponent', requestData);
}
}
}
Рабочий пример можно найти здесь: http://plnkr.co/edit/HDEDDLOeiHEDd7VQaev5?p=preview
Ответ 3
Добавьте общий класс DAL (класс доступа к данным), как этот, и используйте этот класс DAL в своих компонентах.
Добавить индикатор загрузки в качестве сервиса или компонента и использовать для него свои собственные стили.
export class DAL {
private baseUrl: string = environment.apiBaseUrl;
private getConsolidatedPath(path: string) {
if (path.charAt(0) === '/') {
path = path.substr(1);
}
return `${this.baseUrl}/${path}`;
}
private callStack = [];
private startCall() {
this.loadingIndicator.display(true);
this.callStack.push(1);
}
private endCall() {
this.callStack.pop();
if (this.callStack.length === 0) {
this.loadingIndicator.display(false);
}
}
public get(path: string) {
this.startCall();
return this.http.get(this.getConsolidatedPath(path), { headers: this.getHeaders() })
.map(response => response.json())
.catch(e => this.handleError(e))
.finally(() => this.endCall());
}
public put(path: string, data: any) {
this.startCall();
return this.http.put(this.getConsolidatedPath(path), data, { headers: this.getHeaders() })
.map(response => response.json())
.catch(e => this.handleError(e))
.finally(() => this.endCall());
}
public post(path: string, data: any) {
this.startCall();
return this.http.post(this.getConsolidatedPath(path), data, { headers: this.getHeaders() })
.map(response => response.json())
.catch(e => this.handleError(e))
.finally(() => this.endCall());
}
public delete(path: string, data: any) {
this.startCall();
return this.http.delete(this.getConsolidatedPath(path), { body: data, headers: this.getHeaders() })
.map(response => response.json())
.catch(e => this.handleError(e))
.finally(() => this.endCall());
}
constructor(public http: Http, public loadingIndicator: LoadingIndicatorService) {
}
}
Ответ 4
Кроме того @tibbus ответ
Это лучше установить тип "isLoading" Number и сохранить его в обслуживании.
с булевым:
запрос 1 запуск → счетчик на → запрос 2 запуска → запрос 1 концы → счетчик выключения → запрос 2 конца
с номером:
запрос 1 запуск → счетчик вкл. → запрос 2 запуска → запрос 1 концы → запрос 2 конца → выдержка
Сервис
@Injectable()
export class DataService {
constructor(private http: Http) { }
private set fetchCounter(v:number) {
this._fetchCounter = v;
this.isLoadingSource.next(this._fetchCounter > 0)
}
private get fetchCounter() { return this._fetchCounter };
private _fetchCounter:number = 0;
private isLoadingSource = new Subject<boolean>();
public isLoading = this.isLoadingSource.asObservable();
public getData() {
this.fetchCounter++;
return this.http.get('http://jsonplaceholder.typicode.com/posts/2')
.map(r => {
this.fetchCounter--;
return r;
});
}
}
Вам просто нужно подписаться на isLoading из любого из ваших компонентов.