Angular 4 在 2 个不相关的组件之间传递数据

IT技术 javascript angular typescript
2021-02-11 16:48:11

我有一个关于在 Angular 中传递数据的问题。

首先,我没有结构 <parent><child [data]=parent.data></child></parent>

我的结构是

<container>
  <navbar>
    <summary></summary>
    <child-summary><child-summary>
  </navbar>
  <content></content>
</container>

所以,在<summary />我有一个选择发送值到<child-summary /><content />

OnSelect 方法可以很好地使用(更改)内部<summary />组件。

因此,我尝试使用@Input,@Output@EventEmitter指令,但我不知道如何将事件作为组件的 @Input 检索,除非继续使用父/子模式。我创建的所有示例都具有组件之间的关系。

编辑: BehaviorSubject 不工作的例子(所有连接到 API 的服务运行良好,只有 observable 在启动时被触发,但在 select 值改变时不会触发)

shared service = company.service.ts(用于检索公司数据)

import { Injectable } from '@angular/core';
import { Headers, Http, Response } from '@angular/http';
import { BehaviorSubject } from 'rxjs/BehaviorSubject';

import { Observable } from 'rxjs/Observable';
import 'rxjs/add/operator/toPromise';

@Injectable()
export class SrvCompany {

    private accountsNumber = new BehaviorSubject<string[]>([]);
    currentAccountsNumber = this.accountsNumber.asObservable();

    changeMessage(accountsNumber: string[]) {
        this.accountsNumber.next(accountsNumber);
    }

    private _companyUrl = 'api/tiers/';

    constructor(private http: Http) { }

    getSociete(): Promise<Response> {
        let url = this._companyUrl;
        return this.http.get(url).toPromise();
    }
}

invoice.component.ts(“孩子”)

import { Component, OnInit, Input } from '@angular/core';
import { Headers, Http, Response } from '@angular/http';

import { SrvInvoice } from './invoice.service';
import { SrvCompany } from '../company/company.service';

@Component({
    selector: 'invoice',
    templateUrl: 'tsScripts/invoice/invoice.html',
    providers: [SrvInvoice, SrvCompany]
})

export class InvoiceComponent implements OnInit  {

    invoice: any;

    constructor(private srvInvoice: SrvInvoice, private srvCompany: SrvCompany)
    {

    }

    ngOnInit(): void {
        //this.getInvoice("F001");

        // Invoice data is linked to accounts number from company.
        this.srvCompany.currentAccountsNumber.subscribe(accountsNumber => {
            console.log(accountsNumber);
            if (accountsNumber.length > 0) {
                this.srvInvoice.getInvoice(accountsNumber).then(data => this.invoice = data.json());
            }
        });
    }

    //getInvoice(id: any) {
    //    this.srvInvoice.getInvoice(id).then(data => this.invoice = data.json());
    //}
}

company.component.ts(触发“父”)

import { Component, Inject, OnInit, Input } from '@angular/core';
import { Headers, Http, Response } from '@angular/http';

import { SrvCompany } from './company.service';

@Component({
    selector: 'company',
    templateUrl: 'tsScripts/company/company.html',
    providers: [SrvCompany]    
})

export class CompanyComponent implements OnInit {

    societes: any[];    
    soc: Response[]; // debug purpose
    selectedSociete: any;

    ville: any;
    ref: any;
    cp: any;
    accountNumber: any[];

    constructor(private srvSociete: SrvCompany)
    {

    }

    ngOnInit(): void {
        this.getSocietes();
    }

    getSocietes(): void {

        this.srvSociete.getSociete()
            .then(data => this.societes = data.json())
            .then(data => this.selectItem(this.societes[0].Id));
    }

    selectItem(value: any) {
        this.selectedSociete = this.societes.filter((item: any) => item.Id === value)[0];
        this.cp = this.selectedSociete.CodePostal;
        this.ville = this.selectedSociete.Ville;
        this.ref = this.selectedSociete.Id;
        this.accountNumber = this.selectedSociete.Accounts;
        console.log(this.accountNumber);
        this.srvSociete.changeMessage(this.accountNumber);
    }
}
5个回答

这是您想要使用共享服务的情况,因为您的组件被构造为兄弟姐妹和孙辈。这是一个视频示例,我创建了一个关于解决这个确切问题的组件之间共享数据的视频

首先在服务中创建一个 BehaviorSubject

import { Injectable } from '@angular/core';
import { BehaviorSubject } from 'rxjs/BehaviorSubject';
@Injectable()
export class DataService {

  private messageSource = new BehaviorSubject("default message");
  currentMessage = this.messageSource.asObservable();

  constructor() { }

  changeMessage(message: string) {
    this.messageSource.next(message)
  }

}

然后将此服务注入到每个组件中并订阅可观察对象。

import { Component, OnInit } from '@angular/core';
import { DataService } from "../data.service";
@Component({
  selector: 'app-parent',
  template: `
    {{message}}
  `,
  styleUrls: ['./sibling.component.css']
})
export class ParentComponent implements OnInit {

  message:string;

  constructor(private data: DataService) { }

  ngOnInit() {
    this.data.currentMessage.subscribe(message => this.message = message)
  }

}

您可以从任一组件更改值,即使您没有父/子关系,该值也会更新。

import { Component, OnInit } from '@angular/core';
import { DataService } from "../data.service";
@Component({
  selector: 'app-sibling',
  template: `
    {{message}}
    <button (click)="newMessage()">New Message</button>
  `,
  styleUrls: ['./sibling.component.css']
})
export class SiblingComponent implements OnInit {

  message:string;

  constructor(private data: DataService) { }

  ngOnInit() {
    this.data.currentMessage.subscribe(message => this.message = message)
  }

  newMessage() {
    this.data.changeMessage("Hello from Sibling")
  }

}
当您从 selectItem 控制台日志时,您是否获得了预期值?看起来您的订阅模式是正确的。
2021-03-15 16:48:11
好的,经过多次研究,我已经两次将 SrvCompany 声明为提供者,所以我没有一个这样的单例......感谢您的帮助。
2021-03-15 16:48:11
我使用了这个例子,在两个组件的提供者中定义了数据服务,但它也不起作用。但是,当我从组件中的提供者中删除服务并将它们放入主module提供者中时,一切都会相应地进行。希望能帮助到你!
2021-03-20 16:48:11
是的,我按预期有一个包含 2 个对象的数组。并且有疑问,我在 company.component 的 changeMes​​sage 中对数组进行了硬编码。但同样,我在 changeMes​​sage 中表现良好,但接下来没有触发任何事情。
2021-03-21 16:48:11
我已经尝试使用 rxjs/Subject 而不是 rxjs/BehaviorSubject 作为 angular 的文档示例,但它最糟糕(没有被触发)。
2021-04-14 16:48:11
<my-astronaut />在<mission-control />的模板中被引用,所以有相关的组件。
2021-03-20 16:48:11
它的例子但概念是一样的。尝试使用服务,如果您在此处遇到问题评论
2021-04-08 16:48:11

对此有两种解决方案。

  1. 这可以通过使用 observable 的共享服务来完成。

  2. 您可以为此使用 ngrx/store。这类似于 Redux arch。您将从州获取数据。

这是在两个独立组件之间共享数据的最简单示例,使用事件发射器和服务

https://stackoverflow.com/a/44858648/8300620

当您提到非相关组件时,我会假设它们没有任何父组件。如果假设不正确,请随时阅读我的另一个答案,其中解决了两种情况

因此,由于没有共同的父级,我们可以使用可注入的服务。在这种情况下,只需在组件中注入服务并订阅其事件。

(就像下图所示 -取自这里- 除了我们将在两个组件中注入服务)

在组件 Angular 中注入服务

该文档很好地解释了如何创建和注册可注入服务

那么在页面刷新时,所有数据都会丢失吗?即使页面刷新,如何保留数据?
2021-04-04 16:48:11