1

使用排序标题实现AngularMaterial表我在控制台中收到此错误:

错误类型错误:无法设置未定义的属性“排序”

和终端中的这个错误:

错误 TS2740:“MatTableDataSource”类型缺少“Employee[]”类型中的以下属性:长度、弹出、推送、concat 等 24 个。

错误 TS2322:类型 'MatSort' 不可分配给类型 '(compareFn?: (a: Employee, b: Employee) => number) => Employee[]'。

类型 'MatSort' 不提供与签名 '(compareFn?: (a: Employee, b: Employee) => number): Employee[]' 的匹配。

我无法从我这里得到它想要的东西。我将不胜感激任何提示!

我使用了 AngularMaterial 文档中的代码,但数据接口除外。这可能是一个问题吗?

员工列表组件


import { Component, OnInit, ViewChild } from '@angular/core';
import { MatSort, MatTableDataSource } from '@angular/material';
import { AngularFirestore } from '@angular/fire/firestore';
import { EmployeeService } from '../shared/employee.service';
import { Employee } from '../shared/employee';

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

  displayedColumns: string[] = ['fullname', 'position', 'mobile'];
  dataSource;

  @ViewChild(MatSort) sort: MatSort;

  constructor(private service: EmployeeService) { }

  ngOnInit() {
    this.service.getEmployees().subscribe(employees => {
      this.dataSource = new MatTableDataSource(employees);
      console.log(this.dataSource);
    });
    this.dataSource.sort = this.sort;

  }

}


员工列表 HTML


<table mat-table [dataSource]="dataSource" matSort class="mat-elevation-z8">

  <!-- Position Column -->
  <ng-container matColumnDef="fullname">
    <th mat-header-cell *matHeaderCellDef mat-sort-header> Full Name </th>
    <td mat-cell *matCellDef="let element"> {{element.fullname}} </td>
  </ng-container>

  <!-- Name Column -->
  <ng-container matColumnDef="position">
    <th mat-header-cell *matHeaderCellDef mat-sort-header> Position </th>
    <td mat-cell *matCellDef="let element"> {{element.position}} </td>
  </ng-container>

  <!-- Weight Column -->
  <ng-container matColumnDef="mobile">
    <th mat-header-cell *matHeaderCellDef mat-sort-header> Mobile </th>
    <td mat-cell *matCellDef="let element"> {{element.mobile}} </td>
  </ng-container>

  <tr mat-header-row *matHeaderRowDef="displayedColumns"></tr>
  <tr mat-row *matRowDef="let row; columns: displayedColumns;"></tr>
</table>


员工服务


getEmployees() {
    return this.firestore.collection<Employee>('employees')
      .snapshotChanges()
      .pipe(
        map(actions => actions.map(a => {
          const data = a.payload.doc.data();
          const id = a.payload.doc.id;
          return { id, ...data } as Employee;
       })
        )
      );
}

员工类


export class Employee {
  id: string;
  fullname: string;
  empCode: string;
  position: string;
  mobile: string;
}


4

2 回答 2

1

您正在尝试从订阅外部访问数据,它应该在订阅内部。

在此之后,您将遇到this.sortngOnInit钩子中未定义的问题,这是因为您试图访问尚未绘制的 HTML 元素。(垫排序)

有一个生命周期钩子可以用来确保视图已准备好被@ViewChild 访问,这是 ngAfterViewInit,仅在绘制 HTML 时触发。

尝试像这样构建。


import { Component, AfterViewInit, ViewChild } from '@angular/core';
import { MatSort, MatTableDataSource } from '@angular/material';
import { AngularFirestore } from '@angular/fire/firestore';
import { EmployeeService } from '../shared/employee.service';
import { Employee } from '../shared/employee';

@Component({
  selector: 'app-employee-list',
  templateUrl: './employee-list.component.html',
  styleUrls: ['./employee-list.component.scss']
})
export class EmployeeListComponent implements AfterViewInit{

  displayedColumns: string[] = ['fullname', 'position', 'mobile'];
  dataSource;

  @ViewChild(MatSort) sort: MatSort;

  constructor(private service: EmployeeService) { }

  ngAfterViewInit(): void
  {
    this.service.getEmployees().subscribe(employees => {
      this.dataSource = new MatTableDataSource(employees);
      if (this.sort) // check it is defined.
      {
          this.dataSource.sort = this.sort;
      }
    });
  }

}

文档AfterViewInit 生命周期//更多细节

于 2019-05-18T10:54:54.490 回答
0

总体而言,您的代码看起来不错,但是这一行

this.dataSource.sort = this.sort;

位置不对,应该放在subscribe方块里面。

原因_ERROR TypeError: Cannot set property 'sort' of undefined

您正在将您的同步代码与异步混合,我的意思是您的subscribe代码只会在返回 API 的响应时执行(1 到 2 秒),但 this.dataSource.sort = this.sort;不会等待响应,因为它是同步代码,它会继续执行不管任何HTTP请求和您的dataSource对象都会undefined如此,因此您需要将其转移到异步部分。

您的代码将是这样的:

ngOnInit() {
    this.service.getEmployees().subscribe(employees => {
      this.dataSource = new MatTableDataSource(employees);
      console.log(this.dataSource);
      this.dataSource.sort = this.sort; <- note this line 

   });   
}
于 2019-05-18T10:51:44.043 回答