redoc/lib/services/menu.service.ts

216 lines
6.4 KiB
TypeScript
Raw Normal View History

2016-05-07 10:54:44 +03:00
'use strict';
import { Injectable, EventEmitter } from '@angular/core';
2016-11-23 02:23:32 +03:00
import { BehaviorSubject } from 'rxjs/BehaviorSubject';
import { ScrollService, INVIEW_POSITION } from './scroll.service';
import { Hash } from './hash.service';
2016-10-23 20:18:42 +03:00
import { SpecManager } from '../utils/spec-manager';
import { SchemaHelper, MenuCategory } from './schema-helper.service';
2016-11-23 02:23:32 +03:00
import { AppStateService } from './app-state.service';
import { LazyTasksService } from '../shared/components/LazyFor/lazy-for';
2016-05-07 10:54:44 +03:00
const CHANGE = {
NEXT : 1,
BACK : -1,
INITIAL : 0
};
@Injectable()
export class MenuService {
changed: EventEmitter<any> = new EventEmitter();
2016-11-23 02:23:32 +03:00
ready: BehaviorSubject<boolean> = new BehaviorSubject(false);
categories: Array<MenuCategory>;
2016-06-13 20:54:24 +03:00
activeCatIdx: number = 0;
activeMethodIdx: number = -1;
2016-11-23 02:23:32 +03:00
constructor(
private hash:Hash,
private tasks: LazyTasksService,
private scrollService: ScrollService,
private appState: AppStateService,
specMgr:SpecManager
) {
2016-05-07 10:54:44 +03:00
this.hash = hash;
this.categories = SchemaHelper.buildMenuTree(specMgr.schema);
2016-05-07 10:54:44 +03:00
scrollService.scroll.subscribe((evt) => {
this.scrollUpdate(evt.isScrolledDown);
});
this.changeActive(CHANGE.INITIAL);
2016-10-23 20:18:42 +03:00
this.hash.value.subscribe((hash) => {
2016-11-23 02:23:32 +03:00
if (hash == undefined) return;
this.setActiveByHash(hash);
if (!this.tasks.empty) {
this.tasks.start(this.activeCatIdx, this.activeMethodIdx);
this.scrollService.setStickElement(this.getCurrentMethodEl());
2016-11-24 16:47:14 +03:00
if (hash) this.scrollToActive();
2016-11-24 16:29:29 +03:00
this.appState.stopLoading();
2016-11-23 02:23:32 +03:00
} else {
2016-11-24 16:47:14 +03:00
if (hash) this.scrollToActive();
2016-11-23 02:23:32 +03:00
}
2016-05-09 22:55:16 +03:00
});
2016-05-07 10:54:44 +03:00
}
2016-11-23 02:23:32 +03:00
get activeMethodPtr() {
let cat = this.categories[this.activeCatIdx];
let ptr = null;
if (cat && cat.methods.length) {
let mtd = cat.methods[this.activeMethodIdx];
ptr = mtd && mtd.pointer || null;
}
return ptr;
}
2016-05-07 10:54:44 +03:00
scrollUpdate(isScrolledDown) {
let stable = false;
while(!stable) {
let $activeMethodHost = this.getCurrentMethodEl();
if (!$activeMethodHost) return;
var elementInViewPos = this.scrollService.getElementPos($activeMethodHost);
if(isScrolledDown) {
//&& elementInViewPos === INVIEW_POSITION.BELLOW
let $nextEl = this.getRelativeCatOrItem(1);
2016-11-23 02:23:32 +03:00
if (!$nextEl) return;
let nextInViewPos = this.scrollService.getElementPos($nextEl, true);
if (elementInViewPos === INVIEW_POSITION.BELLOW && nextInViewPos === INVIEW_POSITION.ABOVE) {
stable = this.changeActive(CHANGE.NEXT);
continue;
}
2016-05-07 10:54:44 +03:00
}
if(!isScrolledDown && elementInViewPos === INVIEW_POSITION.ABOVE ) {
stable = this.changeActive(CHANGE.BACK);
continue;
}
stable = true;
}
}
getRelativeCatOrItem(offset: number = 0) {
let ptr, cat;
cat = this.categories[this.activeCatIdx];
if (cat.methods.length === 0) {
ptr = null;
cat = this.categories[this.activeCatIdx + Math.sign(offset)] || cat;
} else {
let cat = this.categories[this.activeCatIdx];
let idx = this.activeMethodIdx + offset;
if ((idx >= cat.methods.length - 1) || idx < 0) {
cat = this.categories[this.activeCatIdx + Math.sign(offset)] || cat;
idx = offset > 0 ? -1 : cat.methods.length - 1;
}
ptr = cat.methods[idx] && cat.methods[idx].pointer;
}
return this.getMethodElByPtr(ptr, cat.id);
}
2016-05-07 10:54:44 +03:00
getCurrentMethodEl() {
return this.getMethodElByPtr(this.activeMethodPtr,
2016-07-26 12:03:15 +03:00
this.categories[this.activeCatIdx].id);
2016-05-07 10:54:44 +03:00
}
getMethodElByPtr(ptr, section) {
let selector = ptr ? `[pointer="${ptr}"][section="${section}"]` : `[section="${section}"]`;
2016-05-07 10:54:44 +03:00
return document.querySelector(selector);
}
getMethodElByOperId(operationId) {
let selector =`[operation-id="${operationId}"]`;
return document.querySelector(selector);
}
activate(catIdx, methodIdx) {
2016-11-24 16:29:29 +03:00
if (catIdx < 0) return;
2016-05-07 10:54:44 +03:00
let menu = this.categories;
menu[this.activeCatIdx].active = false;
if (menu[this.activeCatIdx].methods.length) {
if (this.activeMethodIdx >= 0) {
menu[this.activeCatIdx].methods[this.activeMethodIdx].active = false;
}
}
this.activeCatIdx = catIdx;
this.activeMethodIdx = methodIdx;
menu[catIdx].active = true;
let currentItem;
if (menu[catIdx].methods.length && (methodIdx > -1)) {
currentItem = menu[catIdx].methods[methodIdx];
currentItem.active = true;
}
2016-05-17 19:38:12 +03:00
this.changed.next({cat: menu[catIdx], item: currentItem});
2016-05-07 10:54:44 +03:00
}
_calcActiveIndexes(offset) {
let menu = this.categories;
let catCount = menu.length;
if (!catCount) return [0, -1];
2016-05-07 10:54:44 +03:00
let catLength = menu[this.activeCatIdx].methods.length;
let resMethodIdx = this.activeMethodIdx + offset;
let resCatIdx = this.activeCatIdx;
if (resMethodIdx > catLength - 1) {
resCatIdx++;
resMethodIdx = -1;
}
if (resMethodIdx < -1) {
let prevCatIdx = --resCatIdx;
catLength = menu[Math.max(prevCatIdx, 0)].methods.length;
resMethodIdx = catLength - 1;
}
if (resCatIdx > catCount - 1) {
resCatIdx = catCount - 1;
resMethodIdx = catLength - 1;
}
if (resCatIdx < 0) {
resCatIdx = 0;
resMethodIdx = 0;
}
return [resCatIdx, resMethodIdx];
}
changeActive(offset = 1) {
let [catIdx, methodIdx] = this._calcActiveIndexes(offset);
this.activate(catIdx, methodIdx);
return (methodIdx === 0 && catIdx === 0);
}
scrollToActive() {
this.scrollService.scrollTo(this.getCurrentMethodEl());
}
2016-11-23 02:23:32 +03:00
setActiveByHash(hash) {
if (!hash) {
return;
}
let catIdx, methodIdx;
2016-05-07 10:54:44 +03:00
hash = hash.substr(1);
let namespace = hash.split('/')[0];
let ptr = decodeURIComponent(hash.substr(namespace.length + 1));
2016-11-23 02:23:32 +03:00
if (namespace === 'section' || namespace === 'tag') {
2016-07-26 12:03:15 +03:00
let sectionId = ptr.split('/')[0];
2016-11-23 02:23:32 +03:00
catIdx = this.categories.findIndex(cat => cat.id === namespace + '/' + sectionId);
let cat = this.categories[catIdx];
2016-07-26 12:03:15 +03:00
ptr = ptr.substr(sectionId.length) || null;
2016-11-23 02:23:32 +03:00
methodIdx = cat.methods.findIndex(method => method.pointer === ptr);
2016-10-30 18:58:06 +03:00
} else {
2016-11-23 02:23:32 +03:00
catIdx = this.categories.findIndex(cat => {
if (!cat.methods.length) return false;
methodIdx = cat.methods.findIndex(method => method.operationId === ptr || method.pointer === ptr);
if (methodIdx >= 0) {
return true;
} else {
return false;
}
});
2016-05-07 10:54:44 +03:00
}
2016-11-23 02:23:32 +03:00
this.activate(catIdx, methodIdx);
2016-05-07 10:54:44 +03:00
}
}