Maison > Article > interface Web > Passez l'injecteur manuel à la fonction toSignal pour éviter une erreur d'injection de contexte extérieur
L'entrée de signal requise ne peut pas être utilisée dans le constructeur ou l'initialiseur de champ car la valeur n'est pas disponible. Pour accéder à la valeur, ma solution consiste à observer le changement en vigueur du signal, à faire une requête HTTP au serveur et à définir la valeur du signal. Il y a de nombreuses discussions sur la non-utilisation de l'effet, et je dois trouver d'autres solutions pour le supprimer.
Les entrées de signal requises sont accessibles dans les méthodes de cycle de vie ngOnInit et ngOnChanges. Cependant, toSignal génère des erreurs car elles sont en dehors du contexte d'injection. Il peut être corrigé de deux manières :
import { Component, effect, inject, Injector, input, signal } from '@angular/core'; import { getPerson, Person } from './star-war.api'; import { StarWarPersonComponent } from './star-war-person.component'; @Component({ selector: 'app-star-war', standalone: true, imports: [StarWarPersonComponent], template: ` <p>Jedi Id: {{ jedi() }}</p> <app-star-war-person [person]="fighter()" kind="Jedi Fighter" />`, }) export class StarWarComponent { // required signal input jedi = input.required<number>(); injector = inject(Injector); fighter = signal<Person | undefined>(undefined); constructor() { effect((OnCleanup) => { const sub = getPerson(this.jedi(), this.injector) .subscribe((result) => this.fighter.set(result)); OnCleanup(() => sub.unsubscribe()); }); } }
Les changements de code sont les suivants :
export type Person = { name: string; height: string; mass: string; hair_color: string; skin_color: string; eye_color: string; gender: string; films: string[]; }
import { HttpClient } from "@angular/common/http"; import { inject, Injectable } from "@angular/core"; import { catchError, Observable, of, tap } from "rxjs"; import { Person } from "./person.type"; const URL = 'https://swapi.dev/api/people'; @Injectable({ providedIn: 'root' }) export class StarWarService { private readonly http = inject(HttpClient); getData(id: number): Observable<Person | undefined> { return this.http.get<Person>(`${URL}/${id}`).pipe( tap((data) => console.log('data', data)), catchError((err) => { console.error(err); return of(undefined); })); } }
Créez un StarWarService avec une méthode getData pour appeler l'API StarWar afin de récupérer une personne. Le résultat est un Observable d'une personne ou indéfini.
import { Component, input } from '@angular/core'; @Component({ selector: 'app-star-war', standalone: true, template: ` <p>Jedi Id: {{ jedi() }}</p> <p>Sith Id: {{ sith() }}</p> `, }) export class StarWarComponent implements OnInit { // required signal input jedi = input.required<number>(); // required signal input sith = input.required<number>(); ngOnInit(): void {} }
Les Jedi et les Sith sont des entrées de signal requises ; par conséquent, je ne peux pas les utiliser dans le constructeur ni appeler toSignal avec le service pour initialiser les champs.
J'implémente l'interface OnInit et accède aux deux entrées de signal dans la méthode ngOnInit.
import { Component, VERSION } from '@angular/core'; import { StarWarComponent } from './star-war.component'; @Component({ selector: 'app-root', standalone: true, imports: [StarWarComponent], template: ` <app-star-war [jedi]="1" [sith]="4" /> <app-star-war [jedi]="10" [sith]="44" />`, }) export class App {}
Le composant App a deux instances de StarWarComponent. L'identifiant Jedi de la première instance est 1 et l'identifiant de la deuxième instance est 10. Les identifiants Sith des instances sont respectivement 4 et 44.
export class StarWarComponent implements OnInit { // required signal input jedi = input.required<number>(); starWarService = inject(StarWarService); injector = inject(Injector); light!: Signal<Person | undefined>; }
Dans le composant StarWarComponent, j'injecte le StarWarService et l'injecteur du composant. De plus, je déclare un signal lumineux pour stocker le résultat renvoyé par la fonction toSignal.
interface ToSignalOptions<T> { initialValue?: unknown; requireSync?: boolean; injector?: Injector; manualCleanup?: boolean; rejectErrors?: boolean; equal?: ValueEqualityFn<T>; }
L'option ToSignalOptions a une propriété d'injecteur. Lors de l'utilisation de la fonction toSignal en dehors du contexte d'injection, je peux passer l'injecteur du composant à l'option.
export class StarWarComponent implements OnInit { // required signal input jedi = input.required<number>(); starWarService = inject(StarWarService); injector = inject(Injector); light!: Signal<Person | undefined>; ngOnInit(): void { this.light = toSignal(this.starWarService.getData(this.jedi()), { injector: this.injector }); } }
Dans la méthode ngOnInit, j'appelle le service pour obtenir un Observable, et j'utilise la fonction toSignal pour créer un signal. Le deuxième argument est une option avec l'injecteur du composant.
<app-star-war-person [person]="light()" kind="Jedi Fighter" />
Ensuite, je transmets le signal lumineux au composant StarWarPersonComponent pour afficher les détails d'un combattant Jedi.
export class StarWarComponent implements OnInit { // required signal input sith = input.required<number>(); starWarService = inject(StarWarService); injector = inject(Injector); evil!: Signal<Person | undefined>; ngOnInit(): void { // this also works runInInjectionContext(this.injector, () => { this.evil = toSignal(this.starWarService.getData(this.sith())); }) } }
Je déclare un signal maléfique pour stocker le résultat renvoyé par la fonction toSignal. Le premier argument de runInInjectionContext est l’injecteur du composant. Le deuxième argument est une fonction de rappel qui exécute la fonction toSignal et attribue la personne à la variable maléfique.
<app-star-war-person [person]="evil()" kind="Sith Lord" />
Ensuite, je transmets le signal maléfique au composant StarWarPersonComponent pour afficher les détails du Seigneur Sith.
Si un composant nécessite des entrées de signal, je peux accéder aux valeurs de ngOnInit ou ngOnChanges pour effectuer des requêtes HTTP ou d'autres opérations. Ensuite, je n'ai pas besoin de créer un effet pour regarder les signaux requis et appeler le backend.
Cela conclut la 33e journée du défi Ironman.
Références :
Ce qui précède est le contenu détaillé de. pour plus d'informations, suivez d'autres articles connexes sur le site Web de PHP en chinois!