UNPKG

39 kBJavaScriptView Raw
1import PerfectScrollbar from 'perfect-scrollbar';
2import ResizeObserver from 'resize-observer-polyfill';
3import { Subject, fromEvent } from 'rxjs';
4import { auditTime, takeUntil } from 'rxjs/operators';
5import { PLATFORM_ID } from '@angular/core';
6import { isPlatformBrowser } from '@angular/common';
7import { NgZone, Inject, Optional, ElementRef, Directive, Input, Output, EventEmitter, KeyValueDiffers } from '@angular/core';
8import { Geometry, Position } from './perfect-scrollbar.interfaces';
9import { PERFECT_SCROLLBAR_CONFIG, PerfectScrollbarConfig, PerfectScrollbarEvents } from './perfect-scrollbar.interfaces';
10export class PerfectScrollbarDirective {
11 constructor(zone, differs, elementRef, platformId, defaults) {
12 this.zone = zone;
13 this.differs = differs;
14 this.elementRef = elementRef;
15 this.platformId = platformId;
16 this.defaults = defaults;
17 this.instance = null;
18 this.ro = null;
19 this.timeout = null;
20 this.animation = null;
21 this.configDiff = null;
22 this.ngDestroy = new Subject();
23 this.disabled = false;
24 this.psScrollY = new EventEmitter();
25 this.psScrollX = new EventEmitter();
26 this.psScrollUp = new EventEmitter();
27 this.psScrollDown = new EventEmitter();
28 this.psScrollLeft = new EventEmitter();
29 this.psScrollRight = new EventEmitter();
30 this.psYReachEnd = new EventEmitter();
31 this.psYReachStart = new EventEmitter();
32 this.psXReachEnd = new EventEmitter();
33 this.psXReachStart = new EventEmitter();
34 }
35 ngOnInit() {
36 if (!this.disabled && isPlatformBrowser(this.platformId)) {
37 const config = new PerfectScrollbarConfig(this.defaults);
38 config.assign(this.config); // Custom configuration
39 this.zone.runOutsideAngular(() => {
40 this.instance = new PerfectScrollbar(this.elementRef.nativeElement, config);
41 });
42 if (!this.configDiff) {
43 this.configDiff = this.differs.find(this.config || {}).create();
44 this.configDiff.diff(this.config || {});
45 }
46 this.zone.runOutsideAngular(() => {
47 this.ro = new ResizeObserver(() => {
48 this.update();
49 });
50 if (this.elementRef.nativeElement.children[0]) {
51 this.ro.observe(this.elementRef.nativeElement.children[0]);
52 }
53 this.ro.observe(this.elementRef.nativeElement);
54 });
55 this.zone.runOutsideAngular(() => {
56 PerfectScrollbarEvents.forEach((eventName) => {
57 const eventType = eventName.replace(/([A-Z])/g, (c) => `-${c.toLowerCase()}`);
58 fromEvent(this.elementRef.nativeElement, eventType)
59 .pipe(auditTime(20), takeUntil(this.ngDestroy))
60 .subscribe((event) => {
61 this[eventName].emit(event);
62 });
63 });
64 });
65 }
66 }
67 ngOnDestroy() {
68 if (isPlatformBrowser(this.platformId)) {
69 this.ngDestroy.next();
70 this.ngDestroy.complete();
71 if (this.ro) {
72 this.ro.disconnect();
73 }
74 if (this.timeout && typeof window !== 'undefined') {
75 window.clearTimeout(this.timeout);
76 }
77 this.zone.runOutsideAngular(() => {
78 if (this.instance) {
79 this.instance.destroy();
80 }
81 });
82 this.instance = null;
83 }
84 }
85 ngDoCheck() {
86 if (!this.disabled && this.configDiff && isPlatformBrowser(this.platformId)) {
87 const changes = this.configDiff.diff(this.config || {});
88 if (changes) {
89 this.ngOnDestroy();
90 this.ngOnInit();
91 }
92 }
93 }
94 ngOnChanges(changes) {
95 if (changes['disabled'] && !changes['disabled'].isFirstChange() && isPlatformBrowser(this.platformId)) {
96 if (changes['disabled'].currentValue !== changes['disabled'].previousValue) {
97 if (changes['disabled'].currentValue === true) {
98 this.ngOnDestroy();
99 }
100 else if (changes['disabled'].currentValue === false) {
101 this.ngOnInit();
102 }
103 }
104 }
105 }
106 ps() {
107 return this.instance;
108 }
109 update() {
110 if (typeof window !== 'undefined') {
111 if (this.timeout) {
112 window.clearTimeout(this.timeout);
113 }
114 this.timeout = window.setTimeout(() => {
115 if (!this.disabled && this.configDiff) {
116 try {
117 this.zone.runOutsideAngular(() => {
118 if (this.instance) {
119 this.instance.update();
120 }
121 });
122 }
123 catch (error) {
124 // Update can be finished after destroy so catch errors
125 }
126 }
127 }, 0);
128 }
129 }
130 geometry(prefix = 'scroll') {
131 return new Geometry(this.elementRef.nativeElement[prefix + 'Left'], this.elementRef.nativeElement[prefix + 'Top'], this.elementRef.nativeElement[prefix + 'Width'], this.elementRef.nativeElement[prefix + 'Height']);
132 }
133 position(absolute = false) {
134 if (!absolute && this.instance) {
135 return new Position(this.instance.reach.x || 0, this.instance.reach.y || 0);
136 }
137 else {
138 return new Position(this.elementRef.nativeElement.scrollLeft, this.elementRef.nativeElement.scrollTop);
139 }
140 }
141 scrollable(direction = 'any') {
142 const element = this.elementRef.nativeElement;
143 if (direction === 'any') {
144 return element.classList.contains('ps--active-x') ||
145 element.classList.contains('ps--active-y');
146 }
147 else if (direction === 'both') {
148 return element.classList.contains('ps--active-x') &&
149 element.classList.contains('ps--active-y');
150 }
151 else {
152 return element.classList.contains('ps--active-' + direction);
153 }
154 }
155 scrollTo(x, y, speed) {
156 if (!this.disabled) {
157 if (y == null && speed == null) {
158 this.animateScrolling('scrollTop', x, speed);
159 }
160 else {
161 if (x != null) {
162 this.animateScrolling('scrollLeft', x, speed);
163 }
164 if (y != null) {
165 this.animateScrolling('scrollTop', y, speed);
166 }
167 }
168 }
169 }
170 scrollToX(x, speed) {
171 this.animateScrolling('scrollLeft', x, speed);
172 }
173 scrollToY(y, speed) {
174 this.animateScrolling('scrollTop', y, speed);
175 }
176 scrollToTop(offset, speed) {
177 this.animateScrolling('scrollTop', (offset || 0), speed);
178 }
179 scrollToLeft(offset, speed) {
180 this.animateScrolling('scrollLeft', (offset || 0), speed);
181 }
182 scrollToRight(offset, speed) {
183 const left = this.elementRef.nativeElement.scrollWidth -
184 this.elementRef.nativeElement.clientWidth;
185 this.animateScrolling('scrollLeft', left - (offset || 0), speed);
186 }
187 scrollToBottom(offset, speed) {
188 const top = this.elementRef.nativeElement.scrollHeight -
189 this.elementRef.nativeElement.clientHeight;
190 this.animateScrolling('scrollTop', top - (offset || 0), speed);
191 }
192 scrollToElement(element, offset, speed) {
193 if (typeof element === 'string') {
194 element = this.elementRef.nativeElement.querySelector(element);
195 }
196 if (element) {
197 const elementPos = element.getBoundingClientRect();
198 const scrollerPos = this.elementRef.nativeElement.getBoundingClientRect();
199 if (this.elementRef.nativeElement.classList.contains('ps--active-x')) {
200 const currentPos = this.elementRef.nativeElement['scrollLeft'];
201 const position = elementPos.left - scrollerPos.left + currentPos;
202 this.animateScrolling('scrollLeft', position + (offset || 0), speed);
203 }
204 if (this.elementRef.nativeElement.classList.contains('ps--active-y')) {
205 const currentPos = this.elementRef.nativeElement['scrollTop'];
206 const position = elementPos.top - scrollerPos.top + currentPos;
207 this.animateScrolling('scrollTop', position + (offset || 0), speed);
208 }
209 }
210 }
211 animateScrolling(target, value, speed) {
212 if (this.animation) {
213 window.cancelAnimationFrame(this.animation);
214 this.animation = null;
215 }
216 if (!speed || typeof window === 'undefined') {
217 this.elementRef.nativeElement[target] = value;
218 }
219 else if (value !== this.elementRef.nativeElement[target]) {
220 let newValue = 0;
221 let scrollCount = 0;
222 let oldTimestamp = performance.now();
223 let oldValue = this.elementRef.nativeElement[target];
224 const cosParameter = (oldValue - value) / 2;
225 const step = (newTimestamp) => {
226 scrollCount += Math.PI / (speed / (newTimestamp - oldTimestamp));
227 newValue = Math.round(value + cosParameter + cosParameter * Math.cos(scrollCount));
228 // Only continue animation if scroll position has not changed
229 if (this.elementRef.nativeElement[target] === oldValue) {
230 if (scrollCount >= Math.PI) {
231 this.animateScrolling(target, value, 0);
232 }
233 else {
234 this.elementRef.nativeElement[target] = newValue;
235 // On a zoomed out page the resulting offset may differ
236 oldValue = this.elementRef.nativeElement[target];
237 oldTimestamp = newTimestamp;
238 this.animation = window.requestAnimationFrame(step);
239 }
240 }
241 };
242 window.requestAnimationFrame(step);
243 }
244 }
245}
246PerfectScrollbarDirective.decorators = [
247 { type: Directive, args: [{
248 selector: '[perfectScrollbar]',
249 exportAs: 'ngxPerfectScrollbar'
250 },] }
251];
252PerfectScrollbarDirective.ctorParameters = () => [
253 { type: NgZone },
254 { type: KeyValueDiffers },
255 { type: ElementRef },
256 { type: Object, decorators: [{ type: Inject, args: [PLATFORM_ID,] }] },
257 { type: undefined, decorators: [{ type: Optional }, { type: Inject, args: [PERFECT_SCROLLBAR_CONFIG,] }] }
258];
259PerfectScrollbarDirective.propDecorators = {
260 disabled: [{ type: Input }],
261 config: [{ type: Input, args: ['perfectScrollbar',] }],
262 psScrollY: [{ type: Output }],
263 psScrollX: [{ type: Output }],
264 psScrollUp: [{ type: Output }],
265 psScrollDown: [{ type: Output }],
266 psScrollLeft: [{ type: Output }],
267 psScrollRight: [{ type: Output }],
268 psYReachEnd: [{ type: Output }],
269 psYReachStart: [{ type: Output }],
270 psXReachEnd: [{ type: Output }],
271 psXReachStart: [{ type: Output }]
272};
273//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"perfect-scrollbar.directive.js","sourceRoot":"","sources":["../../../../projects/lib/src/lib/perfect-scrollbar.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import PerfectScrollbar from 'perfect-scrollbar';\n\nimport ResizeObserver from 'resize-observer-polyfill';\n\nimport { Subject, fromEvent } from 'rxjs';\nimport { auditTime, takeUntil } from 'rxjs/operators';\n\nimport { PLATFORM_ID } from '@angular/core';\nimport { isPlatformBrowser } from '@angular/common';\nimport { NgZone, Inject, Optional, ElementRef, Directive,\n  OnInit, DoCheck, OnChanges, OnDestroy, Input, Output, EventEmitter,\n  SimpleChanges, KeyValueDiffer, KeyValueDiffers } from '@angular/core';\n\nimport { Geometry, Position } from './perfect-scrollbar.interfaces';\n\nimport { PERFECT_SCROLLBAR_CONFIG, PerfectScrollbarConfig, PerfectScrollbarConfigInterface,\n  PerfectScrollbarEvent, PerfectScrollbarEvents } from './perfect-scrollbar.interfaces';\n\n@Directive({\n  selector: '[perfectScrollbar]',\n  exportAs: 'ngxPerfectScrollbar'\n})\nexport class PerfectScrollbarDirective implements OnInit, OnDestroy, DoCheck, OnChanges {\n  private instance: PerfectScrollbar | null = null;\n\n  private ro: ResizeObserver | null = null;\n\n  private timeout: number | null = null;\n  private animation: number | null = null;\n\n  private configDiff: KeyValueDiffer<string, any> | null = null;\n\n  private readonly ngDestroy: Subject<void> = new Subject();\n\n  @Input() disabled: boolean = false;\n\n  @Input('perfectScrollbar') config?: PerfectScrollbarConfigInterface;\n\n  @Output() psScrollY: EventEmitter<any> = new EventEmitter<any>();\n  @Output() psScrollX: EventEmitter<any> = new EventEmitter<any>();\n\n  @Output() psScrollUp: EventEmitter<any> = new EventEmitter<any>();\n  @Output() psScrollDown: EventEmitter<any> = new EventEmitter<any>();\n  @Output() psScrollLeft: EventEmitter<any> = new EventEmitter<any>();\n  @Output() psScrollRight: EventEmitter<any> = new EventEmitter<any>();\n\n  @Output() psYReachEnd: EventEmitter<any> = new EventEmitter<any>();\n  @Output() psYReachStart: EventEmitter<any> = new EventEmitter<any>();\n  @Output() psXReachEnd: EventEmitter<any> = new EventEmitter<any>();\n  @Output() psXReachStart: EventEmitter<any> = new EventEmitter<any>();\n\n  constructor(private zone: NgZone, private differs: KeyValueDiffers,\n    public elementRef: ElementRef, @Inject(PLATFORM_ID) private platformId: Object,\n    @Optional() @Inject(PERFECT_SCROLLBAR_CONFIG) private defaults: PerfectScrollbarConfigInterface) {}\n\n  ngOnInit(): void {\n    if (!this.disabled && isPlatformBrowser(this.platformId)) {\n      const config = new PerfectScrollbarConfig(this.defaults);\n\n      config.assign(this.config); // Custom configuration\n\n      this.zone.runOutsideAngular(() => {\n        this.instance = new PerfectScrollbar(this.elementRef.nativeElement, config);\n      });\n\n      if (!this.configDiff) {\n        this.configDiff = this.differs.find(this.config || {}).create();\n\n        this.configDiff.diff(this.config || {});\n      }\n\n      this.zone.runOutsideAngular(() => {\n        this.ro = new ResizeObserver(() => {\n          this.update();\n        });\n\n        if (this.elementRef.nativeElement.children[0]) {\n          this.ro.observe(this.elementRef.nativeElement.children[0]);\n        }\n\n        this.ro.observe(this.elementRef.nativeElement);\n      });\n\n      this.zone.runOutsideAngular(() => {\n        PerfectScrollbarEvents.forEach((eventName: PerfectScrollbarEvent) => {\n          const eventType = eventName.replace(/([A-Z])/g, (c) => `-${c.toLowerCase()}`);\n\n          fromEvent<Event>(this.elementRef.nativeElement, eventType)\n            .pipe(\n              auditTime(20),\n              takeUntil(this.ngDestroy)\n            )\n            .subscribe((event: Event) => {\n              this[eventName].emit(event);\n            });\n        });\n      });\n    }\n  }\n\n  ngOnDestroy(): void {\n    if (isPlatformBrowser(this.platformId)) {\n      this.ngDestroy.next();\n      this.ngDestroy.complete();\n\n      if (this.ro) {\n        this.ro.disconnect();\n      }\n\n      if (this.timeout && typeof window !== 'undefined') {\n        window.clearTimeout(this.timeout);\n      }\n\n      this.zone.runOutsideAngular(() => {\n        if (this.instance) {\n          this.instance.destroy();\n        }\n      });\n\n      this.instance = null;\n    }\n  }\n\n  ngDoCheck(): void {\n    if (!this.disabled && this.configDiff && isPlatformBrowser(this.platformId)) {\n      const changes = this.configDiff.diff(this.config || {});\n\n      if (changes) {\n        this.ngOnDestroy();\n\n        this.ngOnInit();\n      }\n    }\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes['disabled'] && !changes['disabled'].isFirstChange() && isPlatformBrowser(this.platformId)) {\n      if (changes['disabled'].currentValue !== changes['disabled'].previousValue) {\n        if (changes['disabled'].currentValue === true) {\n         this.ngOnDestroy();\n        } else if (changes['disabled'].currentValue === false) {\n          this.ngOnInit();\n        }\n      }\n    }\n  }\n\n  public ps(): PerfectScrollbar | null {\n    return this.instance;\n  }\n\n  public update(): void {\n    if (typeof window !== 'undefined') {\n      if (this.timeout) {\n        window.clearTimeout(this.timeout);\n      }\n\n      this.timeout = window.setTimeout(() => {\n        if (!this.disabled && this.configDiff) {\n          try {\n            this.zone.runOutsideAngular(() => {\n              if (this.instance) {\n                this.instance.update();\n              }\n            });\n          } catch (error) {\n            // Update can be finished after destroy so catch errors\n          }\n        }\n      }, 0);\n    }\n  }\n\n  public geometry(prefix: string = 'scroll'): Geometry {\n    return new Geometry(\n      this.elementRef.nativeElement[prefix + 'Left'],\n      this.elementRef.nativeElement[prefix + 'Top'],\n      this.elementRef.nativeElement[prefix + 'Width'],\n      this.elementRef.nativeElement[prefix + 'Height']\n    );\n  }\n\n  public position(absolute: boolean = false): Position {\n    if (!absolute && this.instance) {\n      return new Position(\n        this.instance.reach.x || 0,\n        this.instance.reach.y || 0\n      );\n    } else {\n      return new Position(\n        this.elementRef.nativeElement.scrollLeft,\n        this.elementRef.nativeElement.scrollTop\n      );\n    }\n  }\n\n  public scrollable(direction: string = 'any'): boolean {\n    const element = this.elementRef.nativeElement;\n\n    if (direction === 'any') {\n      return element.classList.contains('ps--active-x') ||\n        element.classList.contains('ps--active-y');\n    } else if (direction === 'both') {\n      return element.classList.contains('ps--active-x') &&\n        element.classList.contains('ps--active-y');\n    } else {\n      return element.classList.contains('ps--active-' + direction);\n    }\n  }\n\n  public scrollTo(x: number, y?: number, speed?: number): void {\n    if (!this.disabled) {\n      if (y == null && speed == null) {\n        this.animateScrolling('scrollTop', x, speed);\n      } else {\n        if (x != null) {\n          this.animateScrolling('scrollLeft', x, speed);\n        }\n\n        if (y != null) {\n          this.animateScrolling('scrollTop', y, speed);\n        }\n      }\n    }\n  }\n\n  public scrollToX(x: number, speed?: number): void {\n    this.animateScrolling('scrollLeft', x, speed);\n  }\n\n  public scrollToY(y: number, speed?: number): void {\n    this.animateScrolling('scrollTop', y, speed);\n  }\n\n  public scrollToTop(offset?: number, speed?: number): void {\n    this.animateScrolling('scrollTop', (offset || 0), speed);\n  }\n\n  public scrollToLeft(offset?: number, speed?: number): void {\n    this.animateScrolling('scrollLeft', (offset || 0), speed);\n  }\n\n  public scrollToRight(offset?: number, speed?: number): void {\n    const left = this.elementRef.nativeElement.scrollWidth -\n      this.elementRef.nativeElement.clientWidth;\n\n    this.animateScrolling('scrollLeft', left - (offset || 0), speed);\n  }\n\n  public scrollToBottom(offset?: number, speed?: number): void {\n    const top = this.elementRef.nativeElement.scrollHeight -\n      this.elementRef.nativeElement.clientHeight;\n\n    this.animateScrolling('scrollTop', top - (offset || 0), speed);\n  }\n\n  public scrollToElement(element: HTMLElement | string, offset?: number, speed?: number): void {\n    if (typeof element === 'string') {\n      element = this.elementRef.nativeElement.querySelector(element) as HTMLElement;\n    }\n\n    if (element) {\n      const elementPos = element.getBoundingClientRect();\n\n      const scrollerPos = this.elementRef.nativeElement.getBoundingClientRect();\n\n      if (this.elementRef.nativeElement.classList.contains('ps--active-x')) {\n        const currentPos = this.elementRef.nativeElement['scrollLeft'];\n\n        const position = elementPos.left - scrollerPos.left + currentPos;\n\n        this.animateScrolling('scrollLeft', position + (offset || 0), speed);\n      }\n\n      if (this.elementRef.nativeElement.classList.contains('ps--active-y')) {\n        const currentPos = this.elementRef.nativeElement['scrollTop'];\n\n        const position = elementPos.top - scrollerPos.top + currentPos;\n\n        this.animateScrolling('scrollTop', position + (offset || 0), speed);\n      }\n    }\n  }\n\n  private animateScrolling(target: string, value: number, speed?: number): void {\n    if (this.animation) {\n      window.cancelAnimationFrame(this.animation);\n\n      this.animation = null;\n    }\n\n    if (!speed || typeof window === 'undefined') {\n      this.elementRef.nativeElement[target] = value;\n    } else if (value !== this.elementRef.nativeElement[target]) {\n      let newValue = 0;\n      let scrollCount = 0;\n\n      let oldTimestamp = performance.now();\n      let oldValue = this.elementRef.nativeElement[target];\n\n      const cosParameter = (oldValue - value) / 2;\n\n      const step = (newTimestamp: number) => {\n        scrollCount += Math.PI / (speed / (newTimestamp - oldTimestamp));\n\n        newValue = Math.round(value + cosParameter + cosParameter * Math.cos(scrollCount));\n\n        // Only continue animation if scroll position has not changed\n        if (this.elementRef.nativeElement[target] === oldValue) {\n          if (scrollCount >= Math.PI) {\n            this.animateScrolling(target, value, 0);\n          } else {\n            this.elementRef.nativeElement[target] = newValue;\n\n            // On a zoomed out page the resulting offset may differ\n            oldValue = this.elementRef.nativeElement[target];\n\n            oldTimestamp = newTimestamp;\n\n            this.animation = window.requestAnimationFrame(step);\n          }\n        }\n      };\n\n      window.requestAnimationFrame(step);\n    }\n  }\n}\n"]}
\No newline at end of file