Search code examples
angularangular-materialrefresh

Angular + Material - How to refresh a data source (mat-table)


I am using a mat-table to list the content of the users chosen languages. They can also add new languages using dialog panel. After they added a language and returned back. I want my datasource to refresh to show the changes they made.

I initialize the datastore by getting user data from a service and passing that into a datasource in the refresh method.

Language.component.ts

import { Component, OnInit } from '@angular/core';
import { LanguageModel, LANGUAGE_DATA } from '../../../../models/language.model';
import { LanguageAddComponent } from './language-add/language-add.component';
import { AuthService } from '../../../../services/auth.service';
import { LanguageDataSource } from './language-data-source';
import { LevelbarComponent } from '../../../../directives/levelbar/levelbar.component';
import { DataSource } from '@angular/cdk/collections';
import { Observable } from 'rxjs/Observable';
import 'rxjs/add/observable/of';
import { MatSnackBar, MatDialog } from '@angular/material';

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

  displayedColumns = ['name', 'native', 'code', 'level'];
  teachDS: any;
  user: any;

  constructor(private authService: AuthService, private dialog: MatDialog) { }

  ngOnInit() {
    this.refresh();
  }

  add() {
    this.dialog.open(LanguageAddComponent, {
      data: { user: this.user },
    }).afterClosed().subscribe(result => {
      this.refresh();
    });
  }

  refresh() {
    this.authService.getAuthenticatedUser().subscribe((res) => {
      this.user = res;
      this.teachDS = new LanguageDataSource(this.user.profile.languages.teach);   
    });
  }
}

language-data-source.ts

import {MatPaginator, MatSort} from '@angular/material';
import {DataSource} from '@angular/cdk/collections';
import {Observable} from 'rxjs/Observable';
import 'rxjs/add/observable/merge';
import 'rxjs/add/operator/map';

export class LanguageDataSource extends DataSource<any> {

  constructor(private languages) {
    super();
  }

  connect(): Observable<any> {
    return Observable.of(this.languages);
  }

  disconnect() {
    // No-op
  }

}

So I have tried to call a refresh method where I get the user from the backend again and then I reinitialize the data source. However this does not work, no changes are occurring.


Solution

  • Trigger a change detection by using ChangeDetectorRef in the refresh() method just after receiving the new data, inject ChangeDetectorRef in the constructor and use detectChanges like this:

    import { Component, OnInit, ChangeDetectorRef } from '@angular/core';
    import { LanguageModel, LANGUAGE_DATA } from '../../../../models/language.model';
    import { LanguageAddComponent } from './language-add/language-add.component';
    import { AuthService } from '../../../../services/auth.service';
    import { LanguageDataSource } from './language-data-source';
    import { LevelbarComponent } from '../../../../directives/levelbar/levelbar.component';
    import { DataSource } from '@angular/cdk/collections';
    import { Observable } from 'rxjs/Observable';
    import 'rxjs/add/observable/of';
    import { MatSnackBar, MatDialog } from '@angular/material';
    
    @Component({
      selector: 'app-language',
      templateUrl: './language.component.html',
      styleUrls: ['./language.component.scss']
    })
    export class LanguageComponent implements OnInit {
      displayedColumns = ['name', 'native', 'code', 'level'];
      teachDS: any;
    
      user: any;
    
      constructor(private authService: AuthService, private dialog: MatDialog,
                  private changeDetectorRefs: ChangeDetectorRef) { }
    
      ngOnInit() {
        this.refresh();
      }
    
      add() {
        this.dialog.open(LanguageAddComponent, {
          data: { user: this.user },
        }).afterClosed().subscribe(result => {
          this.refresh();
        });
      }
    
      refresh() {
        this.authService.getAuthenticatedUser().subscribe((res) => {
          this.user = res;
          this.teachDS = new LanguageDataSource(this.user.profile.languages.teach);
          this.changeDetectorRefs.detectChanges();
        });
      }
    }