angular - 在 Angular 页面中显示全局错误消息

标签 angular

我有嵌套在父页面组件中的子组件。目前,我有一个事件发射器,它会向父组件发出错误事件和消息,以便在遇到错误时显示。有没有办法让错误消息冒泡,以便它们显示在所有页面上。我正在考虑使用 appcomponent 文件,但不确定如何处理它。

Child:
@Output() errorEmitter = new EventEmitter();

errorEmission(message: string){
    this.errorEmitter.emit(message);
  }

Parent:
<app-manage (errorEmitter)='displayError($event)'></app-manage>


displayError(message: string) {
    this.errorMessageIsOn = true;
    this.errorMessageString = message;
  }

有没有一种方法可以使其可扩展,这样我就不必为每个页面而只需为每个组件重写此代码?

最佳答案

简单地说,这个想法是将错误与 UI 逻辑完全解耦:

  1. 创建一个通知( toastr )组件,用于订阅错误、警告、信息、成功消息
  2. 创建一个能够发送消息和通知组件以使用它们的服务。

示例notification.service.ts:

import { Injectable } from '@angular/core'
import { BehaviorSubject, Subject } from 'rxjs';

@Injectable({
  providedIn: 'root'
})
export class NotificationService {
  private readonly errorsSubject$ = new Subject<string>();

  public errors$() { 
    return this.errorsSubject$.asObservable();
  }

  public showError(message: string) : void {
    this.errorsSubject$.next(message);
  }
}

示例 notification.component.ts,您的应用程序中应该只有它的一个实例。

import { Component, Input } from "@angular/core";
import { NotificationService } from "./notification.service";

@Component({
  selector: "notification",
  template: `
    <h2 *ngIf="(error$ | async) as error">Hello {{ error }}!</h2>
  `,
  styles: [
    `
      h1 {
        font-family: Lato;
      }
    `
  ]
})
export class NotificationComponent {
  error$ = this.notificationService.errors$();

  constructor(private readonly notificationService: NotificationService) {}
}

也是一个可以发送消息的示例组件,这可以是执行此操作的任何其他组件。

import { Component, Input } from "@angular/core";
import { NotificationService } from "./notification.service";

@Component({
  selector: "hello",
  template: `
    <button (click)="onClick()">Show error</button>
  `,
  styles: [
    `
      button {
        font-family: Lato;
      }
    `
  ]
})
export class HelloComponent {
  @Input() name: string;
  constructor(private readonly notificationService: NotificationService) {}

  onClick(): void {
    this.notificationService.showError(
      `This error has been posted on ${Date.now()}`
    );
  }
}

现在只要您在任何组件中注入(inject)通知服务并通过该服务发送消息,通知组件就能够订阅它并在全局范围内显示它们。这是Stackblitz显示它正在工作。

显然这非常简单,您需要实现的不止于此,但这应该会让您走上正轨。

一项改进是从通知服务中删除可观察到的 error$ ,仅允许通知组件访问它。您可以通过实现内部通知服务来实现这一点,该服务将充当通知服务和通知组件之间的桥梁。你赢了什么?通知服务不再公开 error$ observable,仅公开发送消息的方法。

notification.service.ts

import { Injectable } from "@angular/core";
import { Subject } from "rxjs";
import { NotificationInternalService } from "./notification-internal.service";

@Injectable({
  providedIn: "root"
})
export class NotificationService {
  constructor(private readonly internalService: NotificationInternalService){}

  public showError(message: string): void {
    this.internalService.errorSubject$.next(message);
  }
}

notification-internal.service.ts

import { Injectable } from "@angular/core";
import { Subject } from "rxjs";

@Injectable({
  providedIn: "root"
})
export class NotificationInternalService {
  public errorSubject$ = new Subject<string>();

  public get error$() {
    return this.errorSubject$.asObservable();
  }
}

并且 notification.component.ts 现在引用 notification-internal.service.ts

import { Component, Input } from "@angular/core";
import { NotificationInternalService } from "./notification-internal.service";

@Component({
  selector: "notification",
  template: `
    <h2 *ngIf="(error$ | async) as error">Hello {{ error }}!</h2>
  `,
  styles: [
    `
      h1 {
        font-family: Lato;
      }
    `
  ]
})
export class NotificationComponent {
  error$ = this.service.error$;

  constructor(private readonly service: NotificationInternalService) {}
}

关于angular - 在 Angular 页面中显示全局错误消息,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/58699757/

相关文章:

css - 在 TS 文件上动态应用 CSS 类

angular - typescript 错误 : Property 'app' does not exist on type 'Navigator'

angular - 当应用于 Angular 中的嵌套网格时,Mat-Table 数据源被覆盖

authentication - Auth0 和 Angular2 : how to set callbackURL and catch the token?

javascript - ngOnChanges 不会在 native 或移动网络中触发。 ionic 4

angular - 如何使用 ng-deep 处理动态数据

更改路线后,Angular 4 脚本不起作用

javascript - *ngFor angular 2 when creating tabs - 表达式在检查后发生了变化

angular - 如何在 Angular 6 同步运行的第一个订阅中使用第二个订阅返回值?

javascript - 如何仅在鼠标滚轮滚动时触发