fuse-angular/src/@fuse/directives/scrollbar/scrollbar.directive.ts
2021-05-14 17:17:06 +03:00

468 lines
12 KiB
TypeScript

import { Directive, ElementRef, Input, OnChanges, OnDestroy, OnInit, SimpleChanges } from '@angular/core';
import { Router } from '@angular/router';
import { BooleanInput, coerceBooleanProperty } from '@angular/cdk/coercion';
import { Platform } from '@angular/cdk/platform';
import { fromEvent, Subject } from 'rxjs';
import { debounceTime, takeUntil } from 'rxjs/operators';
import PerfectScrollbar from 'perfect-scrollbar';
import { merge } from 'lodash-es';
import { ScrollbarGeometry, ScrollbarPosition } from '@fuse/directives/scrollbar/scrollbar.types';
/**
* Wrapper directive for the Perfect Scrollbar: https://github.com/mdbootstrap/perfect-scrollbar
*/
@Directive({
selector: '[fuseScrollbar]',
exportAs: 'fuseScrollbar'
})
export class FuseScrollbarDirective implements OnChanges, OnInit, OnDestroy
{
/* eslint-disable @typescript-eslint/naming-convention */
static ngAcceptInputType_fuseScrollbar: BooleanInput;
/* eslint-enable @typescript-eslint/naming-convention */
@Input() fuseScrollbar: boolean = true;
@Input() fuseScrollbarOptions: PerfectScrollbar.Options;
private _animation: number;
private _options: PerfectScrollbar.Options;
private _ps: PerfectScrollbar;
private _unsubscribeAll: Subject<any> = new Subject<any>();
/**
* Constructor
*/
constructor(
private _elementRef: ElementRef,
private _platform: Platform,
private _router: Router
)
{
}
// -----------------------------------------------------------------------------------------------------
// @ Accessors
// -----------------------------------------------------------------------------------------------------
/**
* Getter for _elementRef
*/
get elementRef(): ElementRef
{
return this._elementRef;
}
/**
* Getter for _ps
*/
get ps(): PerfectScrollbar | null
{
return this._ps;
}
// -----------------------------------------------------------------------------------------------------
// @ Lifecycle hooks
// -----------------------------------------------------------------------------------------------------
/**
* On changes
*
* @param changes
*/
ngOnChanges(changes: SimpleChanges): void
{
// Enabled
if ( 'fuseScrollbar' in changes )
{
// Interpret empty string as 'true'
this.fuseScrollbar = coerceBooleanProperty(changes.fuseScrollbar.currentValue);
// If enabled, init the directive
if ( this.fuseScrollbar )
{
this._init();
}
// Otherwise destroy it
else
{
this._destroy();
}
}
// Scrollbar options
if ( 'fuseScrollbarOptions' in changes )
{
// Merge the options
this._options = merge({}, this._options, changes.fuseScrollbarOptions.currentValue);
// Return if not initialized
if ( !this._ps )
{
return;
}
// Destroy and re-init the PerfectScrollbar to update its options
setTimeout(() => {
this._destroy();
});
setTimeout(() => {
this._init();
});
}
}
/**
* On init
*/
ngOnInit(): void
{
// Subscribe to window resize event
fromEvent(window, 'resize')
.pipe(
takeUntil(this._unsubscribeAll),
debounceTime(150)
)
.subscribe(() => {
// Update the PerfectScrollbar
this.update();
});
}
/**
* On destroy
*/
ngOnDestroy(): void
{
this._destroy();
// Unsubscribe from all subscriptions
this._unsubscribeAll.next();
this._unsubscribeAll.complete();
}
// -----------------------------------------------------------------------------------------------------
// @ Public methods
// -----------------------------------------------------------------------------------------------------
/**
* Is enabled
*/
isEnabled(): boolean
{
return this.fuseScrollbar;
}
/**
* Update the scrollbar
*/
update(): void
{
// Return if not initialized
if ( !this._ps )
{
return;
}
// Update the PerfectScrollbar
this._ps.update();
}
/**
* Destroy the scrollbar
*/
destroy(): void
{
this.ngOnDestroy();
}
/**
* Returns the geometry of the scrollable element
*
* @param prefix
*/
geometry(prefix: string = 'scroll'): ScrollbarGeometry
{
return new ScrollbarGeometry(
this._elementRef.nativeElement[prefix + 'Left'],
this._elementRef.nativeElement[prefix + 'Top'],
this._elementRef.nativeElement[prefix + 'Width'],
this._elementRef.nativeElement[prefix + 'Height']);
}
/**
* Returns the position of the scrollable element
*
* @param absolute
*/
position(absolute: boolean = false): ScrollbarPosition
{
let scrollbarPosition;
if ( !absolute && this._ps )
{
scrollbarPosition = new ScrollbarPosition(
this._ps.reach.x || 0,
this._ps.reach.y || 0
);
}
else
{
scrollbarPosition = new ScrollbarPosition(
this._elementRef.nativeElement.scrollLeft,
this._elementRef.nativeElement.scrollTop
);
}
return scrollbarPosition;
}
/**
* Scroll to
*
* @param x
* @param y
* @param speed
*/
scrollTo(x: number, y?: number, speed?: number): void
{
if ( y == null && speed == null )
{
this.animateScrolling('scrollTop', x, speed);
}
else
{
if ( x != null )
{
this.animateScrolling('scrollLeft', x, speed);
}
if ( y != null )
{
this.animateScrolling('scrollTop', y, speed);
}
}
}
/**
* Scroll to X
*
* @param x
* @param speed
*/
scrollToX(x: number, speed?: number): void
{
this.animateScrolling('scrollLeft', x, speed);
}
/**
* Scroll to Y
*
* @param y
* @param speed
*/
scrollToY(y: number, speed?: number): void
{
this.animateScrolling('scrollTop', y, speed);
}
/**
* Scroll to top
*
* @param offset
* @param speed
*/
scrollToTop(offset: number = 0, speed?: number): void
{
this.animateScrolling('scrollTop', offset, speed);
}
/**
* Scroll to bottom
*
* @param offset
* @param speed
*/
scrollToBottom(offset: number = 0, speed?: number): void
{
const top = this._elementRef.nativeElement.scrollHeight - this._elementRef.nativeElement.clientHeight;
this.animateScrolling('scrollTop', top - offset, speed);
}
/**
* Scroll to left
*
* @param offset
* @param speed
*/
scrollToLeft(offset: number = 0, speed?: number): void
{
this.animateScrolling('scrollLeft', offset, speed);
}
/**
* Scroll to right
*
* @param offset
* @param speed
*/
scrollToRight(offset: number = 0, speed?: number): void
{
const left = this._elementRef.nativeElement.scrollWidth - this._elementRef.nativeElement.clientWidth;
this.animateScrolling('scrollLeft', left - offset, speed);
}
/**
* Scroll to element
*
* @param qs
* @param offset
* @param ignoreVisible If true, scrollToElement won't happen if element is already inside the current viewport
* @param speed
*/
scrollToElement(qs: string, offset: number = 0, ignoreVisible: boolean = false, speed?: number): void
{
const element = this._elementRef.nativeElement.querySelector(qs);
if ( !element )
{
return;
}
const elementPos = element.getBoundingClientRect();
const scrollerPos = this._elementRef.nativeElement.getBoundingClientRect();
if ( this._elementRef.nativeElement.classList.contains('ps--active-x') )
{
if ( ignoreVisible && elementPos.right <= (scrollerPos.right - Math.abs(offset)) )
{
return;
}
const currentPos = this._elementRef.nativeElement['scrollLeft'];
const position = elementPos.left - scrollerPos.left + currentPos;
this.animateScrolling('scrollLeft', position + offset, speed);
}
if ( this._elementRef.nativeElement.classList.contains('ps--active-y') )
{
if ( ignoreVisible && elementPos.bottom <= (scrollerPos.bottom - Math.abs(offset)) )
{
return;
}
const currentPos = this._elementRef.nativeElement['scrollTop'];
const position = elementPos.top - scrollerPos.top + currentPos;
this.animateScrolling('scrollTop', position + offset, speed);
}
}
/**
* Animate scrolling
*
* @param target
* @param value
* @param speed
*/
animateScrolling(target: string, value: number, speed?: number): void
{
if ( this._animation )
{
window.cancelAnimationFrame(this._animation);
this._animation = null;
}
if ( !speed || typeof window === 'undefined' )
{
this._elementRef.nativeElement[target] = value;
}
else if ( value !== this._elementRef.nativeElement[target] )
{
let newValue = 0;
let scrollCount = 0;
let oldTimestamp = performance.now();
let oldValue = this._elementRef.nativeElement[target];
const cosParameter = (oldValue - value) / 2;
const step = (newTimestamp: number) => {
scrollCount += Math.PI / (speed / (newTimestamp - oldTimestamp));
newValue = Math.round(value + cosParameter + cosParameter * Math.cos(scrollCount));
// Only continue animation if scroll position has not changed
if ( this._elementRef.nativeElement[target] === oldValue )
{
if ( scrollCount >= Math.PI )
{
this.animateScrolling(target, value, 0);
}
else
{
this._elementRef.nativeElement[target] = newValue;
// On a zoomed out page the resulting offset may differ
oldValue = this._elementRef.nativeElement[target];
oldTimestamp = newTimestamp;
this._animation = window.requestAnimationFrame(step);
}
}
};
window.requestAnimationFrame(step);
}
}
// -----------------------------------------------------------------------------------------------------
// @ Private methods
// -----------------------------------------------------------------------------------------------------
/**
* Initialize
*
* @private
*/
private _init(): void
{
// Return if already initialized
if ( this._ps )
{
return;
}
// Return if on mobile or not on browser
if ( this._platform.ANDROID || this._platform.IOS || !this._platform.isBrowser )
{
this.fuseScrollbar = false;
return;
}
// Initialize the PerfectScrollbar
this._ps = new PerfectScrollbar(this._elementRef.nativeElement, {...this._options});
}
/**
* Destroy
*
* @private
*/
private _destroy(): void
{
// Return if not initialized
if ( !this._ps )
{
return;
}
// Destroy the PerfectScrollbar
this._ps.destroy();
// Clean up
this._ps = null;
}
}