redoc/lib/components/Redoc/redoc.js

200 lines
5.8 KiB
JavaScript
Raw Normal View History

'use strict';
import {RedocComponent, BaseComponent} from '../base';
2015-10-27 20:44:08 +03:00
import SchemaManager from '../../utils/SchemaManager';
2015-12-26 20:44:39 +03:00
2015-10-27 20:44:08 +03:00
import ApiInfo from '../ApiInfo/api-info';
import ApiLogo from '../ApiLogo/api-logo';
2015-10-27 20:44:08 +03:00
import MethodsList from '../MethodsList/methods-list';
2015-11-16 02:10:04 +03:00
import SideMenu from '../SideMenu/side-menu';
2015-12-26 20:44:39 +03:00
import StickySidebar from '../../common/components/StickySidebar/sticky-sidebar';
2015-12-30 02:29:29 +03:00
import OptionsManager from '../../options';
2015-12-30 11:16:36 +03:00
import {redocEvents} from '../../events';
2015-12-26 20:44:39 +03:00
import {ChangeDetectionStrategy} from 'angular2/core';
2015-12-26 20:44:39 +03:00
import {ElementRef} from 'angular2/core';
2015-12-30 11:16:36 +03:00
import {BrowserDomAdapter, bootstrap} from 'angular2/platform/browser';
2015-12-26 20:44:39 +03:00
import detectScollParent from 'scrollparent';
import {isFunction} from 'angular2/src/facade/lang';
let optionNames = new Set(['scrollYOffset', 'disableLazySchemas']);
2016-01-24 20:01:17 +03:00
let dom = new BrowserDomAdapter();
@RedocComponent({
selector: 'redoc',
2015-12-30 02:29:29 +03:00
providers: [SchemaManager, BrowserDomAdapter, OptionsManager],
templateUrl: './lib/components/Redoc/redoc.html',
2015-11-16 02:15:06 +03:00
styleUrls: ['./lib/components/Redoc/redoc.css'],
2015-12-26 20:44:39 +03:00
directives: [ApiInfo, ApiLogo, MethodsList, SideMenu, StickySidebar],
changeDetection: ChangeDetectionStrategy.Default
})
2015-10-27 20:44:08 +03:00
export default class Redoc extends BaseComponent {
2015-12-30 02:29:29 +03:00
constructor(schemaMgr, optionsMgr, elementRef, dom) {
super(schemaMgr);
2015-12-26 20:44:39 +03:00
this.element = elementRef.nativeElement;
2015-12-27 00:12:10 +03:00
this.dom = dom;
2015-12-26 20:44:39 +03:00
let el = this.element;
//parse options (top level component doesn't support inputs)
2015-12-27 00:12:10 +03:00
this.scrollParent = detectScollParent(el);
this.parseOptions();
2015-12-30 02:29:29 +03:00
this.options = Object.assign({}, optionsMgr.options, this.options);
this.normalizeOptions();
optionsMgr.options = this.options;
2015-12-27 00:12:10 +03:00
}
parseOptions() {
let attributesMap = this.dom.attributeMap(this.element);
this.options = {};
Array.from(attributesMap.keys())
//camelCasify
.map(k => ({
attrName: k,
name: k.replace(/-(.)/g, (m, $1) => $1.toUpperCase())
})
)
.filter(option => optionNames.has(option.name))
.forEach(option => {
this.options[option.name] = attributesMap.get(option.attrName);
});
}
2015-12-27 00:12:10 +03:00
normalizeOptions() {
// modify scrollYOffset to always be a function
if (!isFunction(this.options.scrollYOffset)) {
if (isFinite(this.options.scrollYOffset)) {
// if number specified create function that returns this value
let numberOffset = parseFloat(this.options.scrollYOffset);
this.options.scrollYOffset = () => numberOffset;
} else {
// if selector or node function that returns bottom offset of this node
let el = this.options.scrollYOffset;
if (!(el instanceof Node)) {
el = this.dom.query(el);
}
if (!el) {
this.options.scrollYOffset = () => 0;
} else {
this.options.scrollYOffset = () => el.offsetTop + el.offsetHeight;
}
}
}
this.options.disableLazySchemas = (this.options.disableLazySchemas !== null);
}
2015-12-30 11:16:36 +03:00
2016-01-24 20:01:17 +03:00
static showLoadingAnimation() {
if (!dom.query('#redoc-loading-style')) {
let animStyle = dom.createStyleElement(`
redoc.loading {
position: relative;
display: block;
min-height:350px;
}
redoc.loading:before {
content: "Loading...";
font-size: 28px;
text-align: center;
padding-top: 40px;
color: #3F5C9C;
font-weight: bold;
display: block;
position: absolute;
top: 0;
bottom: 0;
left: 0;
right: 0;
background-color: white;
z-index: 9999;
opacity: 1;
transition: all 0.6s ease-out;
}
redoc.loading-remove:before {
opacity: 0;
}
`);
animStyle.id = 'redoc-loading-style';
dom.appendChild(dom.defaultDoc().head, animStyle);
}
let elem = dom.query('redoc');
dom.addClass(elem, 'loading');
2016-01-24 20:01:17 +03:00
}
static hideLoadingAnimation() {
let redocEl = dom.query('redoc');
dom.addClass(redocEl, 'loading-remove');
setTimeout(() => {
dom.removeClass(redocEl, 'loading-remove');
dom.removeClass(redocEl, 'loading');
}, 400);
}
2015-12-30 11:16:36 +03:00
static init(schemaUrl, options) {
2016-01-15 20:01:09 +03:00
if (Redoc.appRef) {
Redoc.dispose();
}
2016-01-24 20:01:17 +03:00
Redoc.showLoadingAnimation();
2016-01-15 12:35:45 +03:00
return SchemaManager.instance().load(schemaUrl)
.then(() => {
(new OptionsManager()).options = options;
return bootstrap(Redoc);
})
.then(
2016-01-15 20:01:09 +03:00
(appRef) => {
2016-01-24 20:01:17 +03:00
Redoc.hideLoadingAnimation();
2016-01-15 20:01:09 +03:00
Redoc.appRef = appRef;
2016-01-15 12:35:45 +03:00
redocEvents.bootstrapped.next();
console.log('ReDoc bootstrapped!');
},
error => {
console.log(error);
throw error;
}
);
2015-12-30 11:16:36 +03:00
}
2016-01-15 12:35:45 +03:00
2016-01-20 18:05:43 +03:00
static autoInit() {
const specUrlAttributeName = 'spec-url';
let dom = new BrowserDomAdapter();
let redocEl = dom.query('redoc');
if (!redocEl) return;
if (dom.hasAttribute(redocEl, specUrlAttributeName)) {
let url = dom.getAttribute(redocEl, specUrlAttributeName);
Redoc.init(url);
}
}
2016-01-15 20:01:09 +03:00
static dispose() {
let dom = new BrowserDomAdapter();
let el = dom.query('redoc');
let elClone;
2016-01-20 17:37:23 +03:00
let parent;
let nextSibling;
if (el) {
parent = el.parentElement;
nextSibling = el.nextElementSibling;
}
2016-01-15 20:01:09 +03:00
elClone = el.cloneNode(false);
2016-01-20 17:37:23 +03:00
if (Redoc.appRef) {
Redoc.appRef.dispose();
Redoc.appRef = null;
2016-01-15 20:01:09 +03:00
2016-01-20 17:37:23 +03:00
// Redoc dispose removes host element, so need to restore it
elClone.innerHTML = 'Loading...';
parent && parent.insertBefore(elClone, nextSibling);
2016-01-20 17:37:23 +03:00
}
2016-01-15 20:01:09 +03:00
}
}
2015-12-30 02:29:29 +03:00
Redoc.parameters = Redoc.parameters.concat([[OptionsManager], [ElementRef], [BrowserDomAdapter]]);
2015-12-26 20:44:39 +03:00
2016-01-15 12:35:45 +03:00
// TODO
2015-12-26 20:44:39 +03:00
// this doesn't work in side-menu.js because of some circular references issue
SideMenu.parameters = SideMenu.parameters.concat([[Redoc]]);