javascript - 共享服务示例 Angular 5

标签 javascript angular angular5 observable

我知道这个问题已经被问过好几次了,但问题是没有人尝试做一些 fiddle 或显示代码结果。这就是我所拥有的,我需要根据其他组件中的值更新其他组件中的值,但这不仅仅是值,我在其他组件中再次调用了函数。 我有一些组件可以访问数据库并更新值,另一方面,我有其他组件可以从服务的数据库中读取这些值。 这是我的代码示例

tasks.service.ts

import { Injectable } from '@angular/core';
import { HttpClient } from '@angular/common/http';
import { Observable } from 'rxjs';
import 'rxjs/add/operator/catch';
import 'rxjs/add/operator/map';
import { environment } from '../../environments/environment';
import { Tasks } from './tasks';

@Injectable()
export class TasksProvider {

    constructor(private http: HttpClient) { }

    createNewTask(name: Name) : Observable<any> {
        return this.http.post(environment.apiUri + 'tasks', { name, finished: false },
         { responseType: 'text' });
    }

    updateTask(id: Id, name: Name, finished: boolean) : Observable<any> {
        return this.http.put(environment.apiUri + 'tasks/' + id, { name, finished },
         { responseType: 'text' });
    }

    getAllTasks(): Observable<Tasks[]> {
        return this.http.get(environment.apiUri + 'tasks')
            .map<any, Tasks[]>(data => data.map(Tasks.fromObject));
    }
}

app.component.html

<app-tasks-list></app-tasks-list>
<app-tasks-add-new></app-tasks-add-new>

正如您所见,我没有子组件,这是我的主要问题

tasks-list.component.ts

import {Component} from '@angular/core';

import { Tasks } from '../services/tasks';
import { TasksProvider } from '../services/tasks.service'; 

@Component({
  selector: 'app-tasks-list',
  templateUrl: './tasks-list.component.html',
  styleUrls: ['./tasks-list.component.scss']
})

export class TasksListComponent {
  tasks: Array<Tasks>;

  constructor(private tasksProvider: TasksProvider) { }

  ngOnInit() { 
    this.getTasksList();
  }

  displayedColumns: string[] = ['id', 'name', 'finished'];
  private getTasksList() {
        this.tasksProvider.getAllTasks()
            .subscribe(tasks => {
                this.tasks = tasks;
            });
  } 

  public updateCheckboxValue(id: number, name: string, event: any){
    this.tasksProvider.updateTask(id, name, event.checked).subscribe(
      result => {},
      () => {
        alert('Something went wrong');
      })
  }
}

tasks-add-new.component.ts

import { Component, OnInit, Inject } from '@angular/core';
import { MatDialog, MatDialogRef, MAT_DIALOG_DATA } from '@angular/material';

import { Tasks } from '../services/tasks';
import { TasksProvider } from '../services/tasks.service'; 

export interface DialogData {
  name: string;
}

@Component({
  selector: 'app-tasks-add-new',
  templateUrl: './tasks-add-new.component.html',
  styleUrls: ['./tasks-add-new.component.scss']
})
export class TasksAddNewComponent implements OnInit {

  ngOnInit() {
  }

  constructor(public dialog: MatDialog, private tasksProvider: TasksProvider) {}

  openDialog(): void {
    const dialogRef = this.dialog.open(TasksAddNewDialog, {
      width: '250px',
      data: {name: this.animal}
    });

    dialogRef.afterClosed().subscribe(result => {
      this.name = result
      this.tasksProvider.createNewTask(this.name).subscribe(
        result => {},
        () => {
          alert('Something went wrong');
        })
  }
}
}
@Component({
  selector: 'tasks-add-new-dialog',
  templateUrl: 'tasks-add-new-dialog.html'
})

export class TasksAddNewDialog {
  constructor(
    public dialogRef: MatDialogRef<TasksAddNewDialog>,
    @Inject(MAT_DIALOG_DATA) public data: DialogData) {}

  onNoClick(): void {
    this.dialogRef.close();
  }
}

现在当我调用 tasks-add-new.component.ts 中的函数时,您会看到,就像

 this.tasksProvider.createNewTask(this.name).subscribe(
        result => {},
        () => {
          alert('Something went wrong');
        })

我需要再次调用tasks-list.component.ts中的函数

  private getTasksList() {
        this.tasksProvider.getAllTasks()
            .subscribe(tasks => {
                this.tasks = tasks;
            });
  } 

有人知道我如何才能做到这一点吗?

最佳答案

其中一种可能的方法是使用主题。

1)在服务上存储任务列表并提供可订阅的主题

private tasks: Array<Task>;
public $tasks: BehaviorSubject<Array<Task>>;

constructor(private http: HttpClient) {
  this.$tasks = new BehaviorSubject([]);
  ...
}

getAllTasks() {
 this.http.get(environment.apiUri + 'tasks')
    .subscribe(data => {
      this.tasks = data;
      this.$tasks.next(this.tasks);
    });
}

updateTask(params) {
  this.http.post(/* params */).subscribe((task) => {
    this.tasks = this.tasks.map(t => t.id !== task.id ? t : task);
    this.$tasks.next(this.tasks);
  });
}

createTask(...) {
  // again, do a request, update this.tasks and call $tasks.next
  ...
}

2)在组件上进行一项服务Subject订阅,而不是多个服务方法 Observable监听器,并在每次服务源发生变化时自动更新组件列表

tasks: Array<Tasks>;

constructor(private tasksProvider: TasksProvider) {
  this.tasksProvider.$tasks.subscribe(tasks => this.tasks = tasks);
}

ngOnInit() { 
  this.tasksProvider.getAllTasks();
}

public updateCheckboxValue(id: number, name: string, event: any){
  this.tasksProvider.updateTask(id, name, event.checked);
}

关于javascript - 共享服务示例 Angular 5,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/51338259/

相关文章:

angular - angular2 中 md-checkbox 的状态

angularjs - 哪些是 angular2 的好的开源数据网格?

javascript - 动画 ngIf 时元素重叠

javascript - 加载页面时 jQuery 对话框不会打开

javascript - jQuery 无法用meteor.js 改变DOM?

php - 使用 jquery ajax 和 PHP 浏览器 'back button' 的工作

angular - 为什么我的 Eclipse 控制台中有多个 Angular 语言服务器选项卡

javascript - 在 Angular 5 中单击时创建按钮微调器

c# - Angular 5 Uncaught( promise 中): Error: StaticInjectorError(AppModule)

javascript - 如何在 Vuejs 2 中声明全局方法/对象?