Reduce thrashing while initializing blockchain scroll position
This commit is contained in:
		
							parent
							
								
									b979594a04
								
							
						
					
					
						commit
						b645ad3fd8
					
				@ -1,4 +1,4 @@
 | 
			
		||||
import { Component, OnInit, OnDestroy, ChangeDetectionStrategy, Input, Output, EventEmitter, HostListener, ChangeDetectorRef, OnChanges, SimpleChanges } from '@angular/core';
 | 
			
		||||
import { Component, OnInit, OnDestroy, ChangeDetectionStrategy, Input, Output, EventEmitter, ChangeDetectorRef, OnChanges, SimpleChanges } from '@angular/core';
 | 
			
		||||
import { firstValueFrom, Subscription } from 'rxjs';
 | 
			
		||||
import { StateService } from '../../services/state.service';
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@ -1,4 +1,4 @@
 | 
			
		||||
import { Component, ElementRef, HostListener, OnInit, OnDestroy, ViewChild, Input, DoCheck } from '@angular/core';
 | 
			
		||||
import { Component, ElementRef, HostListener, OnInit, OnDestroy, ViewChild, Input, ChangeDetectorRef, ChangeDetectionStrategy, AfterViewChecked } from '@angular/core';
 | 
			
		||||
import { Subscription } from 'rxjs';
 | 
			
		||||
import { MarkBlockState, StateService } from '../../services/state.service';
 | 
			
		||||
import { specialBlocks } from '../../app.constants';
 | 
			
		||||
@ -8,8 +8,9 @@ import { BlockExtended } from '../../interfaces/node-api.interface';
 | 
			
		||||
  selector: 'app-start',
 | 
			
		||||
  templateUrl: './start.component.html',
 | 
			
		||||
  styleUrls: ['./start.component.scss'],
 | 
			
		||||
  changeDetection: ChangeDetectionStrategy.OnPush
 | 
			
		||||
})
 | 
			
		||||
export class StartComponent implements OnInit, OnDestroy, DoCheck {
 | 
			
		||||
export class StartComponent implements OnInit, AfterViewChecked, OnDestroy {
 | 
			
		||||
  @Input() showLoadingIndicator = false;
 | 
			
		||||
 | 
			
		||||
  interval = 60;
 | 
			
		||||
@ -42,6 +43,7 @@ export class StartComponent implements OnInit, OnDestroy, DoCheck {
 | 
			
		||||
  pageWidth: number;
 | 
			
		||||
  firstPageWidth: number;
 | 
			
		||||
  minScrollWidth: number;
 | 
			
		||||
  currentScrollWidth: number = null;
 | 
			
		||||
  pageIndex: number = 0;
 | 
			
		||||
  pages: any[] = [];
 | 
			
		||||
  pendingMark: number | null = null;
 | 
			
		||||
@ -49,9 +51,10 @@ export class StartComponent implements OnInit, OnDestroy, DoCheck {
 | 
			
		||||
  lastUpdate: number = 0;
 | 
			
		||||
  lastMouseX: number;
 | 
			
		||||
  velocity: number = 0;
 | 
			
		||||
  mempoolOffset: number = 0;
 | 
			
		||||
  mempoolOffset: number | null = null;
 | 
			
		||||
  mempoolWidth: number = 0;
 | 
			
		||||
  scrollLeft: number = null;
 | 
			
		||||
 | 
			
		||||
  private resizeObserver: ResizeObserver;
 | 
			
		||||
  chainWidth: number = window.innerWidth;
 | 
			
		||||
  menuOpen: boolean = false;
 | 
			
		||||
  menuSliding: boolean = false;
 | 
			
		||||
@ -59,19 +62,12 @@ export class StartComponent implements OnInit, OnDestroy, DoCheck {
 | 
			
		||||
 | 
			
		||||
  constructor(
 | 
			
		||||
    private stateService: StateService,
 | 
			
		||||
    private cd: ChangeDetectorRef,
 | 
			
		||||
  ) {
 | 
			
		||||
    this.isiOS = ['iPhone','iPod','iPad'].includes((navigator as any)?.userAgentData?.platform || navigator.platform);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  ngDoCheck(): void {
 | 
			
		||||
    if (this.pendingOffset != null) {
 | 
			
		||||
      const offset = this.pendingOffset;
 | 
			
		||||
      this.pendingOffset = null;
 | 
			
		||||
      this.addConvertedScrollOffset(offset);
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  ngOnInit() {
 | 
			
		||||
  ngOnInit(): void {
 | 
			
		||||
    this.firstPageWidth = 40 + (this.blockWidth * this.dynamicBlocksAmount);
 | 
			
		||||
    this.blockCounterSubscription = this.stateService.blocks$.subscribe((blocks) => {
 | 
			
		||||
      this.blockCount = blocks.length;
 | 
			
		||||
@ -122,7 +118,7 @@ export class StartComponent implements OnInit, OnDestroy, DoCheck {
 | 
			
		||||
            this.scrollToBlock(scrollToHeight);
 | 
			
		||||
          }
 | 
			
		||||
        }
 | 
			
		||||
        if (!this.tipIsSet || (blockHeight < 0 && !this.mempoolOffset)) {
 | 
			
		||||
        if (!this.tipIsSet || (blockHeight < 0 && this.mempoolOffset == null)) {
 | 
			
		||||
          this.pendingMark = blockHeight;
 | 
			
		||||
        }
 | 
			
		||||
      }
 | 
			
		||||
@ -168,11 +164,41 @@ export class StartComponent implements OnInit, OnDestroy, DoCheck {
 | 
			
		||||
    });
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  onMempoolOffsetChange(offset): void {
 | 
			
		||||
    const delta = offset - this.mempoolOffset;
 | 
			
		||||
  ngAfterViewChecked(): void {
 | 
			
		||||
    if (this.currentScrollWidth !== this.blockchainContainer?.nativeElement?.scrollWidth) {
 | 
			
		||||
      this.currentScrollWidth = this.blockchainContainer?.nativeElement?.scrollWidth;
 | 
			
		||||
      if (this.pendingOffset != null) {
 | 
			
		||||
        const delta = this.pendingOffset - (this.mempoolOffset || 0);
 | 
			
		||||
        this.mempoolOffset = this.pendingOffset;
 | 
			
		||||
        this.currentScrollWidth = this.blockchainContainer?.nativeElement?.scrollWidth;
 | 
			
		||||
        this.pendingOffset = null;
 | 
			
		||||
        this.addConvertedScrollOffset(delta);
 | 
			
		||||
    this.mempoolOffset = offset;
 | 
			
		||||
        this.applyPendingMarkArrow();
 | 
			
		||||
      } else {
 | 
			
		||||
        this.applyScrollLeft();
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  onMempoolOffsetChange(offset): void {
 | 
			
		||||
    this.pendingOffset = offset;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  applyScrollLeft(): void {
 | 
			
		||||
    if (this.blockchainContainer?.nativeElement?.scrollWidth) {
 | 
			
		||||
      let lastScrollLeft = null;
 | 
			
		||||
      while (this.scrollLeft < 0 && this.shiftPagesForward() && lastScrollLeft !== this.scrollLeft) {
 | 
			
		||||
        lastScrollLeft = this.scrollLeft;
 | 
			
		||||
        this.scrollLeft += this.pageWidth;
 | 
			
		||||
      }
 | 
			
		||||
      lastScrollLeft = null;
 | 
			
		||||
      while (this.scrollLeft > this.blockchainContainer.nativeElement.scrollWidth && this.shiftPagesBack() && lastScrollLeft !== this.scrollLeft) {
 | 
			
		||||
        lastScrollLeft = this.scrollLeft;
 | 
			
		||||
        this.scrollLeft -= this.pageWidth;
 | 
			
		||||
      }
 | 
			
		||||
      this.blockchainContainer.nativeElement.scrollLeft = this.scrollLeft;
 | 
			
		||||
    }
 | 
			
		||||
    this.cd.markForCheck();
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  applyPendingMarkArrow(): void {
 | 
			
		||||
@ -191,6 +217,7 @@ export class StartComponent implements OnInit, OnDestroy, DoCheck {
 | 
			
		||||
    window.clearTimeout(this.menuTimeout);
 | 
			
		||||
    this.menuTimeout = window.setTimeout(() => {
 | 
			
		||||
      this.menuSliding = false;
 | 
			
		||||
      this.cd.markForCheck();
 | 
			
		||||
    }, 300);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
@ -200,9 +227,8 @@ export class StartComponent implements OnInit, OnDestroy, DoCheck {
 | 
			
		||||
    this.isMobile = this.chainWidth <= 767.98;
 | 
			
		||||
    let firstVisibleBlock;
 | 
			
		||||
    let offset;
 | 
			
		||||
    if (this.blockchainContainer?.nativeElement != null) {
 | 
			
		||||
    this.pages.forEach(page => {
 | 
			
		||||
        const left = page.offset - this.getConvertedScrollOffset();
 | 
			
		||||
      const left = page.offset - this.getConvertedScrollOffset(this.scrollLeft);
 | 
			
		||||
      const right = left + this.pageWidth;
 | 
			
		||||
      if (left <= 0 && right > 0) {
 | 
			
		||||
        const blockIndex = Math.max(0, Math.floor(left / -this.blockWidth));
 | 
			
		||||
@ -210,14 +236,13 @@ export class StartComponent implements OnInit, OnDestroy, DoCheck {
 | 
			
		||||
        offset = left + (blockIndex * this.blockWidth);
 | 
			
		||||
      }
 | 
			
		||||
    });
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    this.blocksPerPage = Math.ceil(this.chainWidth / this.blockWidth);
 | 
			
		||||
    this.pageWidth = this.blocksPerPage * this.blockWidth;
 | 
			
		||||
    this.minScrollWidth = this.firstPageWidth + (this.pageWidth * 2);
 | 
			
		||||
 | 
			
		||||
    if (firstVisibleBlock != null) {
 | 
			
		||||
      this.scrollToBlock(firstVisibleBlock, offset + (this.isMobile ? this.blockWidth : 0));
 | 
			
		||||
      this.scrollToBlock(firstVisibleBlock, offset);
 | 
			
		||||
    } else {
 | 
			
		||||
      this.updatePages();
 | 
			
		||||
    }
 | 
			
		||||
@ -227,7 +252,7 @@ export class StartComponent implements OnInit, OnDestroy, DoCheck {
 | 
			
		||||
    if (!(event.which > 1 || event.button > 0)) {
 | 
			
		||||
      this.mouseDragStartX = event.clientX;
 | 
			
		||||
      this.resetMomentum(event.clientX);
 | 
			
		||||
      this.blockchainScrollLeftInit = this.blockchainContainer.nativeElement.scrollLeft;
 | 
			
		||||
      this.blockchainScrollLeftInit = this.scrollLeft;
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
  onPointerDown(event: PointerEvent) {
 | 
			
		||||
@ -253,8 +278,8 @@ export class StartComponent implements OnInit, OnDestroy, DoCheck {
 | 
			
		||||
    if (this.mouseDragStartX != null) {
 | 
			
		||||
      this.updateVelocity(event.clientX);
 | 
			
		||||
      this.stateService.setBlockScrollingInProgress(true);
 | 
			
		||||
      this.blockchainContainer.nativeElement.scrollLeft =
 | 
			
		||||
        this.blockchainScrollLeftInit + this.mouseDragStartX - event.clientX;
 | 
			
		||||
      this.scrollLeft = this.blockchainScrollLeftInit + this.mouseDragStartX - event.clientX;
 | 
			
		||||
      this.applyScrollLeft();
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
  @HostListener('document:mouseup', [])
 | 
			
		||||
@ -310,25 +335,31 @@ export class StartComponent implements OnInit, OnDestroy, DoCheck {
 | 
			
		||||
        } else {
 | 
			
		||||
          this.velocity += dv;
 | 
			
		||||
        }
 | 
			
		||||
        this.blockchainContainer.nativeElement.scrollLeft -= displacement;
 | 
			
		||||
        this.scrollLeft -= displacement;
 | 
			
		||||
        this.applyScrollLeft();
 | 
			
		||||
        this.animateMomentum();
 | 
			
		||||
      }
 | 
			
		||||
    });
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  onScroll(e) {
 | 
			
		||||
    if (this.blockchainContainer?.nativeElement?.scrollLeft == null) {
 | 
			
		||||
      return;
 | 
			
		||||
    }
 | 
			
		||||
    this.scrollLeft = this.blockchainContainer?.nativeElement?.scrollLeft;
 | 
			
		||||
    const middlePage = this.pageIndex === 0 ? this.pages[0] : this.pages[1];
 | 
			
		||||
    // compensate for css transform
 | 
			
		||||
    const translation = (this.isMobile ? this.chainWidth * 0.95 : this.chainWidth * 0.5);
 | 
			
		||||
    const backThreshold = middlePage.offset + (this.pageWidth * 0.5) + translation;
 | 
			
		||||
    const forwardThreshold = middlePage.offset - (this.pageWidth * 0.5) + translation;
 | 
			
		||||
    const scrollLeft = this.getConvertedScrollOffset();
 | 
			
		||||
    if (scrollLeft > backThreshold) {
 | 
			
		||||
    this.scrollLeft = this.blockchainContainer.nativeElement.scrollLeft;
 | 
			
		||||
    const offsetScroll = this.getConvertedScrollOffset(this.scrollLeft);
 | 
			
		||||
    if (offsetScroll > backThreshold) {
 | 
			
		||||
      if (this.shiftPagesBack()) {
 | 
			
		||||
        this.addConvertedScrollOffset(-this.pageWidth);
 | 
			
		||||
        this.blockchainScrollLeftInit -= this.pageWidth;
 | 
			
		||||
      }
 | 
			
		||||
    } else if (scrollLeft < forwardThreshold) {
 | 
			
		||||
    } else if (offsetScroll < forwardThreshold) {
 | 
			
		||||
      if (this.shiftPagesForward()) {
 | 
			
		||||
        this.addConvertedScrollOffset(this.pageWidth);
 | 
			
		||||
        this.blockchainScrollLeftInit += this.pageWidth;
 | 
			
		||||
@ -337,10 +368,6 @@ export class StartComponent implements OnInit, OnDestroy, DoCheck {
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  scrollToBlock(height, blockOffset = 0) {
 | 
			
		||||
    if (!this.blockchainContainer?.nativeElement) {
 | 
			
		||||
      setTimeout(() => { this.scrollToBlock(height, blockOffset); }, 50);
 | 
			
		||||
      return;
 | 
			
		||||
    }
 | 
			
		||||
    if (this.isMobile) {
 | 
			
		||||
      blockOffset -= this.blockWidth;
 | 
			
		||||
    }
 | 
			
		||||
@ -348,15 +375,15 @@ export class StartComponent implements OnInit, OnDestroy, DoCheck {
 | 
			
		||||
    const pages = [];
 | 
			
		||||
    this.pageIndex = Math.max(viewingPageIndex - 1, 0);
 | 
			
		||||
    let viewingPage = this.getPageAt(viewingPageIndex);
 | 
			
		||||
    const isLastPage = viewingPage.height < this.blocksPerPage;
 | 
			
		||||
    const isLastPage = viewingPage.height <= 0;
 | 
			
		||||
    if (isLastPage) {
 | 
			
		||||
      this.pageIndex = Math.max(viewingPageIndex - 2, 0);
 | 
			
		||||
      viewingPage = this.getPageAt(viewingPageIndex);
 | 
			
		||||
    }
 | 
			
		||||
    const left = viewingPage.offset - this.getConvertedScrollOffset();
 | 
			
		||||
    const left = viewingPage.offset - this.getConvertedScrollOffset(this.scrollLeft);
 | 
			
		||||
    const blockIndex = viewingPage.height - height;
 | 
			
		||||
    const targetOffset = (this.blockWidth * blockIndex) + left;
 | 
			
		||||
    let deltaOffset = targetOffset - blockOffset;
 | 
			
		||||
    const deltaOffset = targetOffset - blockOffset;
 | 
			
		||||
 | 
			
		||||
    if (isLastPage) {
 | 
			
		||||
      pages.push(this.getPageAt(viewingPageIndex - 2));
 | 
			
		||||
@ -386,6 +413,7 @@ export class StartComponent implements OnInit, OnDestroy, DoCheck {
 | 
			
		||||
    pages.push(this.getPageAt(this.pageIndex + 1));
 | 
			
		||||
    pages.push(this.getPageAt(this.pageIndex + 2));
 | 
			
		||||
    this.pages = pages;
 | 
			
		||||
    this.cd.markForCheck();
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  shiftPagesBack(): boolean {
 | 
			
		||||
@ -439,44 +467,40 @@ export class StartComponent implements OnInit, OnDestroy, DoCheck {
 | 
			
		||||
  blockInViewport(height: number): boolean {
 | 
			
		||||
    const firstHeight = this.pages[0].height;
 | 
			
		||||
    const translation = (this.isMobile ? this.chainWidth * 0.95 : this.chainWidth * 0.5);
 | 
			
		||||
    const firstX = this.pages[0].offset - this.getConvertedScrollOffset() + translation;
 | 
			
		||||
    const firstX = this.pages[0].offset - this.getConvertedScrollOffset(this.scrollLeft) + translation;
 | 
			
		||||
    const xPos = firstX + ((firstHeight - height) * 155);
 | 
			
		||||
    return xPos > -55 && xPos < (this.chainWidth - 100);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  getConvertedScrollOffset(): number {
 | 
			
		||||
  getConvertedScrollOffset(scrollLeft): number {
 | 
			
		||||
    if (this.timeLtr) {
 | 
			
		||||
      return -(this.blockchainContainer?.nativeElement?.scrollLeft || 0) - this.mempoolOffset;
 | 
			
		||||
      return -(scrollLeft || 0) - (this.mempoolOffset || 0);
 | 
			
		||||
    } else {
 | 
			
		||||
      return (this.blockchainContainer?.nativeElement?.scrollLeft || 0) - this.mempoolOffset;
 | 
			
		||||
      return (scrollLeft || 0) - (this.mempoolOffset || 0);
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  setScrollLeft(offset: number): void {
 | 
			
		||||
    if (this.timeLtr) {
 | 
			
		||||
      this.blockchainContainer.nativeElement.scrollLeft = offset - this.mempoolOffset;
 | 
			
		||||
      this.scrollLeft = offset - (this.mempoolOffset || 0);
 | 
			
		||||
    } else {
 | 
			
		||||
      this.blockchainContainer.nativeElement.scrollLeft = offset + this.mempoolOffset;
 | 
			
		||||
      this.scrollLeft = offset + (this.mempoolOffset || 0);
 | 
			
		||||
    }
 | 
			
		||||
    this.applyScrollLeft();
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  addConvertedScrollOffset(offset: number): void {
 | 
			
		||||
    if (!this.blockchainContainer?.nativeElement) {
 | 
			
		||||
      this.pendingOffset = offset;
 | 
			
		||||
      return;
 | 
			
		||||
    }
 | 
			
		||||
    if (this.timeLtr) {
 | 
			
		||||
      this.blockchainContainer.nativeElement.scrollLeft -= offset;
 | 
			
		||||
      this.scrollLeft -= offset;
 | 
			
		||||
    } else {
 | 
			
		||||
      this.blockchainContainer.nativeElement.scrollLeft += offset;
 | 
			
		||||
      this.scrollLeft += offset;
 | 
			
		||||
    }
 | 
			
		||||
    this.applyScrollLeft();
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  ngOnDestroy() {
 | 
			
		||||
    if (this.blockchainContainer?.nativeElement) {
 | 
			
		||||
    // clean up scroll position to prevent caching wrong scroll in Firefox
 | 
			
		||||
    this.setScrollLeft(0);
 | 
			
		||||
    }
 | 
			
		||||
    this.timeLtrSubscription.unsubscribe();
 | 
			
		||||
    this.chainTipSubscription.unsubscribe();
 | 
			
		||||
    this.markBlockSubscription.unsubscribe();
 | 
			
		||||
 | 
			
		||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user