mirror of
https://github.com/richard-loafle/fuse-angular.git
synced 2025-04-03 15:11:37 +00:00
468 lines
12 KiB
TypeScript
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;
|
|
}
|
|
}
|