Angular CDK 虚拟滚动无限循环

标签 angular angular-cdk

我正在尝试使用 CDK 虚拟滚动为项目列表实现无限循环滚动

我到了放弃的地步。我做不到,我怎样才能实现这个 CDK 虚拟滚动?

这是我的代码

import { Component, ViewChild, ChangeDetectionStrategy } from '@angular/core';
import { CdkVirtualScrollViewport } from '@angular/cdk/scrolling';
import { Observable, BehaviorSubject } from 'rxjs';
import { map } from 'rxjs/operators';

@Component({
  selector: 'my-app',
  template: `
   <ng-container *ngIf="infinite | async; let items">

     <cdk-virtual-scroll-viewport itemSize="50" 
                                  (scrolledIndexChange)="nextBatch($event, (items[items.length - 1].index))">
       <div *cdkVirtualFor="let item of items; trackBy: trackByIdx">{{item}}</div>
     </cdk-virtual-scroll-viewport>

   </ng-container>
  `,
  changeDetection: ChangeDetectionStrategy.OnPush
})
export class AppComponent {

  arr = Array.from({ length: 10 }).map((_, i) => `Item #${i}`);

  offset = new BehaviorSubject<any>({ offset: 0 });
  infinite: Observable<any[]>;

  @ViewChild(CdkVirtualScrollViewport) viewPort: CdkVirtualScrollViewport;


  constructor() {
    this.infinite = this.offset.pipe(
      map((n: any) => this.getBatch(n)),
    );
  }

  getBatch({ offset, loadPrev, loadNext, e }) {
    let arr = [...this.arr];
    if (loadPrev) {
      const last = arr[arr.length - 1];
      return [last, ...arr];
    }
    if (loadNext) {
      const first = arr[0];
      return [...arr, first];
    }
    return arr;
  }

  nextBatch(e, offset) {
    const end = this.viewPort.getRenderedRange().end;
    const start = this.viewPort.getRenderedRange().start;
    const total = this.viewPort.getDataLength();

    if (start === 0) {
      this.offset.next({ offset: offset, loadPrev: true, e });
    }

    if (end === total) {
      this.offset.next({ offset: offset, loadNext: true, e });
    }
  }

  trackByIdx(i) {
    return i;
  }
}

Live stackblitz

最佳答案

我们只需要检查滚动条是否滚动到最后一项,然后再滚动回第一项。例如:

app.component.html

<cdk-virtual-scroll-viewport class="example-viewport" [itemSize]="itemSize" (scrolledIndexChange)="onScroll($event)">
    <div *cdkVirtualFor="let item of items" class="example-item">{{item}}</div>
</cdk-virtual-scroll-viewport>

应用程序组件.ts

items = Array.from({ length: 100 }).map((_, i) => `Item #${i}`);
itemSize = 5;
@ViewChild(CdkVirtualScrollViewport) viewPort: CdkVirtualScrollViewport;

onScroll(event) {
    console.log(event);
    if( event >= ((this.items.length-this.itemSize-1)*10) ) {
      this.viewPort.scrollToIndex(0);
    }
}

https://angular-utnzft.stackblitz.io

关于Angular CDK 虚拟滚动无限循环,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/53143204/

相关文章:

Angular 8 形式重置,值无法正常工作

javascript - Angular CDK 拖放 - 是否可以创建圆形边界?

angular - 使用 angular 4 在 mat-table 中添加一个新行

angular - cdkStepLabel - ng-模板 - innerHTML

javascript - 动态附加项目时,masonry 不会将它们布局出来

unit-testing - 如何在 Angular2 中使用 karma-jasmine 对 ng2-translate 进行单元测试

angular - 如何从单元测试中获取/设置 FormGroup 中的 Ionic 2 ionic 输入 FormControl 值?

javascript - 特定跨度的 Angular 切换颜色

html - 滚动闪烁问题 - 使用带有滚动捕捉类型的 cdk-virtual-scroll

在调整屏幕大小之前,Angular cdk-virtual-scroll-viewport 不会采用最大高度