mirror of
https://github.com/Redocly/redoc.git
synced 2024-11-24 09:33:44 +03:00
107 lines
3.0 KiB
TypeScript
107 lines
3.0 KiB
TypeScript
'use strict';
|
|
import { Injectable, EventEmitter } from '@angular/core';
|
|
import { BrowserDomAdapter as DOM } from '../utils/browser-adapter';
|
|
import { OptionsService } from './options.service';
|
|
import { throttle } from '../utils/helpers';
|
|
|
|
export const INVIEW_POSITION = {
|
|
ABOVE : 1,
|
|
BELLOW: -1,
|
|
INVIEW: 0
|
|
};
|
|
|
|
@Injectable()
|
|
export class ScrollService {
|
|
scrollYOffset: any;
|
|
$scrollParent: any;
|
|
scroll = new EventEmitter();
|
|
private prevOffsetY: number;
|
|
private _cancel:any;
|
|
private _savedPosition:number;
|
|
private _stickElement: HTMLElement;
|
|
constructor(private optionsService:OptionsService) {
|
|
this.scrollYOffset = () => optionsService.options.scrollYOffset();
|
|
this.$scrollParent = optionsService.options.$scrollParent || window;
|
|
this.scroll = new EventEmitter();
|
|
this.bind();
|
|
if ('scrollRestoration' in history) {
|
|
history.scrollRestoration = 'manual';
|
|
}
|
|
}
|
|
|
|
scrollY() {
|
|
return (this.$scrollParent.pageYOffset != undefined) ? this.$scrollParent.pageYOffset : this.$scrollParent.scrollTop;
|
|
}
|
|
|
|
/* returns 1 if element if above the view, 0 if in view and -1 below the view */
|
|
getElementPos($el, inverted=false) {
|
|
let scrollYOffset = this.scrollYOffset();
|
|
let mul = inverted ? -1 : 1;
|
|
if (mul*Math.floor($el.getBoundingClientRect().top) > mul*scrollYOffset) {
|
|
return INVIEW_POSITION.ABOVE;
|
|
}
|
|
|
|
if (mul*$el.getBoundingClientRect().bottom <= mul*scrollYOffset) {
|
|
return INVIEW_POSITION.BELLOW;
|
|
}
|
|
return INVIEW_POSITION.INVIEW;
|
|
}
|
|
|
|
scrollToPos(posY: number) {
|
|
if (this.$scrollParent.scrollTo) {
|
|
this.$scrollParent.scrollTo(0, Math.floor(posY));
|
|
} else {
|
|
this.$scrollParent.scrollTop = posY;
|
|
}
|
|
}
|
|
scrollTo($el, offset:number = 0) {
|
|
if (!$el) return;
|
|
// TODO: rewrite this to use offsetTop as more reliable solution
|
|
let subjRect = $el.getBoundingClientRect();
|
|
let posY = this.scrollY() + subjRect.top - this.scrollYOffset() + offset + 1;
|
|
this.scrollToPos(posY);
|
|
return posY;
|
|
}
|
|
|
|
saveScroll() {
|
|
let $el = this._stickElement;
|
|
if (!$el) return;
|
|
let offsetParent = $el.offsetParent;
|
|
this._savedPosition = $el.offsetTop + (<any>offsetParent).offsetTop;
|
|
}
|
|
|
|
setStickElement($el) {
|
|
this._stickElement = $el;
|
|
}
|
|
|
|
restoreScroll() {
|
|
let $el = this._stickElement;
|
|
if (!$el) return;
|
|
let offsetParent = $el.offsetParent;
|
|
let currentPosition = $el.offsetTop + (<any>offsetParent).offsetTop;
|
|
let newY = this.scrollY() + (currentPosition - this._savedPosition);
|
|
this.scrollToPos(newY);
|
|
}
|
|
|
|
relativeScrollPos($el) {
|
|
let subjRect = $el.getBoundingClientRect();
|
|
return -subjRect.top + this.scrollYOffset() - 1;
|
|
}
|
|
|
|
scrollHandler(evt) {
|
|
let isScrolledDown = (this.scrollY() - this.prevOffsetY > 0);
|
|
this.prevOffsetY = this.scrollY();
|
|
this.scroll.next({isScrolledDown, evt});
|
|
}
|
|
|
|
bind() {
|
|
this.prevOffsetY = this.scrollY();
|
|
this._cancel = DOM.onAndCancel(this.$scrollParent, 'scroll',
|
|
throttle((evt) => { this.scrollHandler(evt); }, 100, this));
|
|
}
|
|
|
|
unbind() {
|
|
this._cancel();
|
|
}
|
|
}
|