gpt4 book ai didi

javascript - react Angular Material 数据表

转载 作者:太空狗 更新时间:2023-10-29 18:00:27 26 4
gpt4 key购买 nike

我用这个 ng generate @angular/material:material-table 命令创建了一个 Angular Material 数据表,它给了我以下文件结构:

  • table-datasource.ts
  • table.component.ts
  • table.component.html

这里的想法是在 table-datasource.ts 中完成所有的抓取、排序和分页。默认情况下,数据放置在 table-datasource.ts 内的数组中,但在我的例子中,它来自 ngxs-store,它公开了一个数组的 Observable。 Atm 我有以下实现:

表数据源.ts:

export class TokenTableDataSource extends DataSource<TokenTableItem> {
@Select(TokenTableState.getTokenTableItems) private tokenTableItems$:Observable<TokenTableItem[]>;
totalItems$ = new BehaviorSubject<TokenTableItem[]>([]);

constructor(private paginator: MatPaginator, private sort: MatSort) {
super();
}

/**
* Connect this data source to the table. The table will only update when
* the returned stream emits new items.
* @returns A stream of the items to be rendered.
*/
connect(): Observable<TokenTableItem[]> {
this.tokenTableItems$.subscribe(item => this.totalItems$.next(item));

// init on first connect
if (this.totalItems$.value === undefined) {
this.totalItems$.next([]);
this.paginator.length = this.totalItems$.value.length;
}
// Combine everything that affects the rendered data into one update
// stream for the data-table to consume.
const dataMutations = [
observableOf(this.totalItems$),
this.paginator.page,
this.sort.sortChange
];

return merge(...dataMutations).pipe(
map(() => this.totalItems$.next(this.getPagedData(this.getSortedData([...this.totalItems$.value])))),
mergeMap(() => this.totalItems$)
);
}
...generated paging and sorting methods

表格组件.html:

<div class="mat-elevation-z8">
<table mat-table class="full-width-table" [dataSource]="dataSource" matSort aria-label="Elements">

...multiple columns

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

<mat-paginator #paginator
[length]="this.dataSource.totalItems$.value?.length"
[pageIndex]="pageIndex"
[pageSize]="pageSize"
[pageSizeOptions]="pageSizeOptions"
[showFirstLastButtons]=true
(page)="handlePage($event)">
</mat-paginator>
</div>

table.component.ts:

export class TokenTableComponent implements OnInit {
@ViewChild(MatPaginator) paginator: MatPaginator;
@ViewChild(MatSort) sort: MatSort;
dataSource: TokenTableDataSource;

pageSizeOptions = [5, 10, 20, 40];
pageSize = this.pageSizeOptions[0];
pageIndex = 0;
tableLength = 0;

... colums definition

ngOnInit(): void {
this.dataSource = new TokenTableDataSource(this.paginator, this.sort);
}

public handlePage(pageEvent: PageEvent) {
// do what?
}
}

什么在起作用:

  • 数据呈现正确(通过按钮和 ngxs-store 触发)
  • 我可以对数据进行排序

什么不起作用:

  • 在第一次加载数据时,pageSize 被完全忽略,所有行都显示出来
  • 单击排序或分页元素时,采用当前选定的 pageSize 并呈现此行数。对我来说奇怪的是,这只适用于降序(给定 pageSize 是 10,我选择 5 它导致 5 行但是一旦选择 5 就不可能再次显示超过 5 行)

要求:

  • 我喜欢将所有数据操作封装在 TableDataSource.connect() 之后的想法,所以像 this 这样的解决方案不需要在组件中完成提取的位置。此外,这没有实现排序。
  • 该应用程序使用了一个 ngxs-store,它与 ngrx 非常相似,因此欢迎任何涉及这部分的解决方案。
  • 我还没有弄清楚如何处理 pageEvents,所以我猜测解决方案是在 handlePage() 方法中。

版本:

  • RxJS 6.3.x
  • Angular 7.x
  • ngxs 3.3.x

最佳答案

我想出了如何根据我的要求设置表格。主要的变化是我删除了从 TableDataSource 获取数据的 Observable 并引入了一个 DataService:

export class DataService {
//the @Select is from ngxs but can be anything returning an Observable
@Select(TokenTableState.getTokenTableItems) private tokenTableItems$: Observable<TokenTableViewItem[]>;
private initValue$: BehaviorSubject<TokenTableViewItem[]> = new BehaviorSubject<TokenTableViewItem[]>([]);

getAllItems(): Observable<TokenTableViewItem[]> {
const sources = [this.tokenTableItems$, this.initValue$];
return merge(...sources);
}
}

基本上,该服务从任何 Observable 输入中获取数据,并将其合并到 getAllItems 方法中并赋予初始值。

组件 有这个服务的一个实例:

私有(private)_dataService:数据服务|空;

它在加载方法中将其交给TableDatasource:

private loadData(): any {
this._dataService = new DataService();
this.dataSource = new TokenTableDataSource(
this._dataService,
this.paginator,
this.sort
);
fromEvent(this.filter.nativeElement, 'keyup').subscribe(() => {
if (!this.dataSource) {
return;
}
this.dataSource.filter = this.filter.nativeElement.value;
});
}

我没有在 TableDataSource 中引用 DataService 的原因是 Component 中的分页器需要的长度渲染表(见下文)。

TableDataSource 像这样使用 DataService:

  • connect 方法中,它包含一个包含可能数据突变的数组:

    const dataMutations = [
    this._dataChange,
    this._sort.sortChange,
    this._filterChange,
    this._paginator.page
    ];
  • 数组的 _dataChange 成员通过订阅我们的 DataServicegetAllItems 方法获取它的值:

    this._internalService.getAllItems().subscribe(data => {
    this._dataChange.next(data);
    });
  • dataMutations 用于过滤、排序和返回应显示的数据:

    return merge(...dataMutations).pipe(
    map(() => {
    // Filter data
    this.filteredData = this._dataChange.value
    .slice()
    .filter((item: TokenTableViewItem) => {
    const searchStr = (item.symbol + item.name).toLowerCase();
    return searchStr.indexOf(this.filter.toLowerCase()) !== -1;
    });
    // Sort filtered data
    const sortedData = this.getSortedData(this.filteredData.slice());

    // Grab the page's slice of the filtered sorted data.
    this.renderedData = this.getPagedData(sortedData);
    return this.renderedData;
    })
    );

filterChange 在本地实例中定义

_filterChange = new BehaviorSubject('');

分页排序是通过构造函数从外部触发的

constructor(
public _internalService: DataService,
public _paginator: MatPaginator,
public _sort: MatSort
) {
super();
this._filterChange.subscribe(() => (this._paginator.pageIndex = 0));
}

我还找到了在 component.html 中定义的分页解决方案,如下所示:

<mat-paginator #paginator
[length]="dataSource.filteredData.length"
[pageIndex]="pageIndex"
[pageSize]="pageSize"
[pageSizeOptions]="pageSizeOptions"
[showFirstLastButtons]=true>
</mat-paginator>

并在 component.ts 中设置变量:

pageSizeOptions = [5, 10, 20, 40];
pageSize = this.pageSizeOptions[0];
pageIndex = 0;

完整代码可见this项目和表的实时版本用于 whatsmytoken.com .

关于javascript - react Angular Material 数据表,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/53759963/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com