2018-02-28 12:05:59 -08:00
|
|
|
import { Component, ElementRef, EventEmitter, Input, OnDestroy, Output } from '@angular/core';
|
2018-01-19 14:58:23 +00:00
|
|
|
import { Title, Meta } from '@angular/platform-browser';
|
2017-07-31 15:45:18 +03:00
|
|
|
|
2019-10-11 17:31:26 +02:00
|
|
|
import { asapScheduler, Observable, of, timer } from 'rxjs';
|
|
|
|
import { catchError, observeOn, switchMap, takeUntil, tap } from 'rxjs/operators';
|
2017-03-01 14:30:25 +00:00
|
|
|
|
2018-01-19 14:58:23 +00:00
|
|
|
import { DocumentContents, FILE_NOT_FOUND_ID, FETCHING_ERROR_ID } from 'app/documents/document.service';
|
2017-07-31 15:45:18 +03:00
|
|
|
import { Logger } from 'app/shared/logger.service';
|
2017-04-27 15:32:46 -07:00
|
|
|
import { TocService } from 'app/shared/toc.service';
|
2018-02-28 12:05:59 -08:00
|
|
|
import { ElementsLoader } from 'app/custom-elements/elements-loader';
|
2017-03-01 14:30:25 +00:00
|
|
|
|
|
|
|
|
2017-12-19 01:53:38 +02:00
|
|
|
// Constants
|
|
|
|
export const NO_ANIMATIONS = 'no-animations';
|
|
|
|
|
2017-03-01 14:30:25 +00:00
|
|
|
// Initialization prevents flicker once pre-rendering is on
|
|
|
|
const initialDocViewerElement = document.querySelector('aio-doc-viewer');
|
|
|
|
const initialDocViewerContent = initialDocViewerElement ? initialDocViewerElement.innerHTML : '';
|
|
|
|
|
|
|
|
@Component({
|
|
|
|
selector: 'aio-doc-viewer',
|
2017-04-27 15:32:46 -07:00
|
|
|
template: ''
|
2017-03-01 14:30:25 +00:00
|
|
|
// TODO(robwormald): shadow DOM and emulated don't work here (?!)
|
2020-10-08 16:59:29 +02:00
|
|
|
// encapsulation: ViewEncapsulation.ShadowDom
|
2017-03-01 14:30:25 +00:00
|
|
|
})
|
2018-02-28 12:05:59 -08:00
|
|
|
export class DocViewerComponent implements OnDestroy {
|
2017-11-23 15:04:29 +02:00
|
|
|
// Enable/Disable view transition animations.
|
|
|
|
static animationsEnabled = true;
|
2017-03-01 14:30:25 +00:00
|
|
|
|
|
|
|
private hostElement: HTMLElement;
|
|
|
|
|
2017-07-31 15:45:18 +03:00
|
|
|
private void$ = of<void>(undefined);
|
|
|
|
private onDestroy$ = new EventEmitter<void>();
|
|
|
|
private docContents$ = new EventEmitter<DocumentContents>();
|
|
|
|
|
2017-11-23 15:03:56 +02:00
|
|
|
protected currViewContainer: HTMLElement = document.createElement('div');
|
|
|
|
protected nextViewContainer: HTMLElement = document.createElement('div');
|
2017-07-31 15:45:18 +03:00
|
|
|
|
|
|
|
@Input()
|
|
|
|
set doc(newDoc: DocumentContents) {
|
|
|
|
// Ignore `undefined` values that could happen if the host component
|
|
|
|
// does not initially specify a value for the `doc` input.
|
|
|
|
if (newDoc) {
|
|
|
|
this.docContents$.emit(newDoc);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-11-27 23:06:09 +02:00
|
|
|
// The new document is ready to be inserted into the viewer.
|
|
|
|
// (Embedded components have been loaded and instantiated, if necessary.)
|
|
|
|
@Output() docReady = new EventEmitter<void>();
|
|
|
|
|
|
|
|
// The previous document has been removed from the viewer.
|
|
|
|
// (The leaving animation (if any) has been completed and the node has been removed from the DOM.)
|
|
|
|
@Output() docRemoved = new EventEmitter<void>();
|
|
|
|
|
|
|
|
// The new document has been inserted into the viewer.
|
|
|
|
// (The node has been inserted into the DOM, but the entering animation may still be in progress.)
|
|
|
|
@Output() docInserted = new EventEmitter<void>();
|
|
|
|
|
|
|
|
// The new document has been fully rendered into the viewer.
|
|
|
|
// (The entering animation has been completed.)
|
|
|
|
@Output() docRendered = new EventEmitter<void>();
|
2017-03-13 09:20:09 +00:00
|
|
|
|
2017-03-01 14:30:25 +00:00
|
|
|
constructor(
|
2018-03-06 14:02:25 -08:00
|
|
|
elementRef: ElementRef,
|
|
|
|
private logger: Logger,
|
|
|
|
private titleService: Title,
|
|
|
|
private metaService: Meta,
|
|
|
|
private tocService: TocService,
|
|
|
|
private elementsLoader: ElementsLoader) {
|
2017-03-01 14:30:25 +00:00
|
|
|
this.hostElement = elementRef.nativeElement;
|
|
|
|
// Security: the initialDocViewerContent comes from the prerendered DOM and is considered to be secure
|
|
|
|
this.hostElement.innerHTML = initialDocViewerContent;
|
|
|
|
|
2017-11-23 15:03:56 +02:00
|
|
|
if (this.hostElement.firstElementChild) {
|
|
|
|
this.currViewContainer = this.hostElement.firstElementChild as HTMLElement;
|
|
|
|
}
|
|
|
|
|
2017-07-31 15:45:18 +03:00
|
|
|
this.docContents$
|
2018-03-21 03:40:23 +02:00
|
|
|
.pipe(
|
2019-10-11 17:31:26 +02:00
|
|
|
observeOn(asapScheduler),
|
2018-03-21 03:40:23 +02:00
|
|
|
switchMap(newDoc => this.render(newDoc)),
|
|
|
|
takeUntil(this.onDestroy$),
|
|
|
|
)
|
2017-07-31 15:45:18 +03:00
|
|
|
.subscribe();
|
2017-03-01 14:30:25 +00:00
|
|
|
}
|
|
|
|
|
2017-07-31 15:45:18 +03:00
|
|
|
ngOnDestroy() {
|
|
|
|
this.onDestroy$.emit();
|
2017-03-01 14:30:25 +00:00
|
|
|
}
|
|
|
|
|
2017-11-27 23:06:09 +02:00
|
|
|
/**
|
|
|
|
* Prepare for setting the window title and ToC.
|
|
|
|
* Return a function to actually set them.
|
|
|
|
*/
|
|
|
|
protected prepareTitleAndToc(targetElem: HTMLElement, docId: string): () => void {
|
|
|
|
const titleEl = targetElem.querySelector('h1');
|
2018-03-03 13:12:08 +00:00
|
|
|
const needsToc = !!titleEl && !/no-?toc/i.test(titleEl.className);
|
|
|
|
const embeddedToc = targetElem.querySelector('aio-toc.embedded');
|
2017-11-27 23:06:09 +02:00
|
|
|
|
2020-10-16 21:46:24 +03:00
|
|
|
if (titleEl && needsToc && !embeddedToc) {
|
2018-03-03 13:12:08 +00:00
|
|
|
// Add an embedded ToC if it's needed and there isn't one in the content already.
|
2020-10-16 21:46:24 +03:00
|
|
|
titleEl.insertAdjacentHTML('afterend', '<aio-toc class="embedded"></aio-toc>');
|
2019-01-16 20:21:40 +01:00
|
|
|
} else if (!needsToc && embeddedToc && embeddedToc.parentNode !== null) {
|
2018-03-03 13:12:08 +00:00
|
|
|
// Remove the embedded Toc if it's there and not needed.
|
2019-01-16 20:21:40 +01:00
|
|
|
// We cannot use ChildNode.remove() because of IE11
|
|
|
|
embeddedToc.parentNode.removeChild(embeddedToc);
|
2017-11-27 23:06:09 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return () => {
|
|
|
|
this.tocService.reset();
|
2018-01-10 10:41:15 +00:00
|
|
|
let title: string|null = '';
|
2017-11-27 23:06:09 +02:00
|
|
|
|
|
|
|
// Only create ToC for docs with an `<h1>` heading.
|
|
|
|
// If you don't want a ToC, add "no-toc" class to `<h1>`.
|
|
|
|
if (titleEl) {
|
|
|
|
title = (typeof titleEl.innerText === 'string') ? titleEl.innerText : titleEl.textContent;
|
|
|
|
|
2018-03-03 13:12:08 +00:00
|
|
|
if (needsToc) {
|
2017-11-27 23:06:09 +02:00
|
|
|
this.tocService.genToc(targetElem, docId);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
this.titleService.setTitle(title ? `Angular - ${title}` : 'Angular');
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2017-03-01 14:30:25 +00:00
|
|
|
/**
|
2017-07-31 15:45:18 +03:00
|
|
|
* Add doc content to host element and build it out with embedded components.
|
2017-03-01 14:30:25 +00:00
|
|
|
*/
|
2017-07-31 15:45:18 +03:00
|
|
|
protected render(doc: DocumentContents): Observable<void> {
|
2017-11-27 23:06:09 +02:00
|
|
|
let addTitleAndToc: () => void;
|
|
|
|
|
2018-01-19 14:58:23 +00:00
|
|
|
this.setNoIndex(doc.id === FILE_NOT_FOUND_ID || doc.id === FETCHING_ERROR_ID);
|
|
|
|
|
2018-03-21 03:40:23 +02:00
|
|
|
return this.void$.pipe(
|
2017-11-27 23:06:09 +02:00
|
|
|
// Security: `doc.contents` is always authored by the documentation team
|
|
|
|
// and is considered to be safe.
|
2018-03-21 03:40:23 +02:00
|
|
|
tap(() => this.nextViewContainer.innerHTML = doc.contents || ''),
|
|
|
|
tap(() => addTitleAndToc = this.prepareTitleAndToc(this.nextViewContainer, doc.id)),
|
2018-05-22 17:31:05 +03:00
|
|
|
switchMap(() => this.elementsLoader.loadContainedCustomElements(this.nextViewContainer)),
|
2018-03-21 03:40:23 +02:00
|
|
|
tap(() => this.docReady.emit()),
|
|
|
|
switchMap(() => this.swapViews(addTitleAndToc)),
|
|
|
|
tap(() => this.docRendered.emit()),
|
|
|
|
catchError(err => {
|
2018-01-03 23:42:55 +02:00
|
|
|
const errorMessage = (err instanceof Error) ? err.stack : err;
|
2018-03-12 11:05:36 +00:00
|
|
|
this.logger.error(new Error(`[DocViewer] Error preparing document '${doc.id}': ${errorMessage}`));
|
2017-11-23 15:03:56 +02:00
|
|
|
this.nextViewContainer.innerHTML = '';
|
2018-01-19 14:58:23 +00:00
|
|
|
this.setNoIndex(true);
|
2017-07-31 15:45:18 +03:00
|
|
|
return this.void$;
|
2018-03-21 03:40:23 +02:00
|
|
|
}),
|
|
|
|
);
|
2017-03-01 14:30:25 +00:00
|
|
|
}
|
2017-11-23 15:03:56 +02:00
|
|
|
|
2018-01-19 14:58:23 +00:00
|
|
|
/**
|
|
|
|
* Tell search engine crawlers whether to index this page
|
|
|
|
*/
|
|
|
|
private setNoIndex(val: boolean) {
|
|
|
|
if (val) {
|
|
|
|
this.metaService.addTag({ name: 'robots', content: 'noindex' });
|
|
|
|
} else {
|
|
|
|
this.metaService.removeTag('name="robots"');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-11-23 15:03:56 +02:00
|
|
|
/**
|
|
|
|
* Swap the views, removing `currViewContainer` and inserting `nextViewContainer`.
|
|
|
|
* (At this point all content should be ready, including having loaded and instantiated embedded
|
|
|
|
* components.)
|
2017-11-27 23:06:09 +02:00
|
|
|
*
|
|
|
|
* Optionally, run a callback as soon as `nextViewContainer` has been inserted, but before the
|
|
|
|
* entering animation has been completed. This is useful for work that needs to be done as soon as
|
|
|
|
* the element has been attached to the DOM.
|
2017-11-23 15:03:56 +02:00
|
|
|
*/
|
2018-01-10 10:41:15 +00:00
|
|
|
protected swapViews(onInsertedCb = () => {}): Observable<void> {
|
2017-11-23 15:04:29 +02:00
|
|
|
const raf$ = new Observable<void>(subscriber => {
|
|
|
|
const rafId = requestAnimationFrame(() => {
|
|
|
|
subscriber.next();
|
|
|
|
subscriber.complete();
|
|
|
|
});
|
|
|
|
return () => cancelAnimationFrame(rafId);
|
|
|
|
});
|
|
|
|
|
2017-12-12 15:26:03 +02:00
|
|
|
// Get the actual transition duration (taking global styles into account).
|
|
|
|
// According to the [CSSOM spec](https://drafts.csswg.org/cssom/#serializing-css-values),
|
|
|
|
// `time` values should be returned in seconds.
|
|
|
|
const getActualDuration = (elem: HTMLElement) => {
|
2018-01-10 10:41:15 +00:00
|
|
|
const cssValue = getComputedStyle(elem).transitionDuration || '';
|
2017-12-12 15:26:03 +02:00
|
|
|
const seconds = Number(cssValue.replace(/s$/, ''));
|
|
|
|
return 1000 * seconds;
|
|
|
|
};
|
2020-05-15 10:35:45 +02:00
|
|
|
|
|
|
|
// Some properties are not assignable and thus cannot be animated.
|
|
|
|
// Example methods, readonly and CSS properties:
|
|
|
|
// "length", "parentRule", "getPropertyPriority", "getPropertyValue", "item", "removeProperty", "setProperty"
|
|
|
|
type StringValueCSSStyleDeclaration
|
|
|
|
= Exclude<{ [K in keyof CSSStyleDeclaration]: CSSStyleDeclaration[K] extends string ? K : never }[keyof CSSStyleDeclaration], number>;
|
2017-11-23 15:04:29 +02:00
|
|
|
const animateProp =
|
2020-05-15 10:35:45 +02:00
|
|
|
(elem: HTMLElement, prop: StringValueCSSStyleDeclaration, from: string, to: string, duration = 200) => {
|
2017-12-19 01:53:38 +02:00
|
|
|
const animationsDisabled = !DocViewerComponent.animationsEnabled
|
|
|
|
|| this.hostElement.classList.contains(NO_ANIMATIONS);
|
2017-11-23 15:04:29 +02:00
|
|
|
elem.style.transition = '';
|
2017-12-19 01:53:38 +02:00
|
|
|
return animationsDisabled
|
2018-03-21 03:40:23 +02:00
|
|
|
? this.void$.pipe(tap(() => elem.style[prop] = to))
|
|
|
|
: this.void$.pipe(
|
2017-11-23 15:04:29 +02:00
|
|
|
// In order to ensure that the `from` value will be applied immediately (i.e.
|
|
|
|
// without transition) and that the `to` value will be affected by the
|
|
|
|
// `transition` style, we need to ensure an animation frame has passed between
|
|
|
|
// setting each style.
|
2018-03-21 03:40:23 +02:00
|
|
|
switchMap(() => raf$), tap(() => elem.style[prop] = from),
|
|
|
|
switchMap(() => raf$), tap(() => elem.style.transition = `all ${duration}ms ease-in-out`),
|
2019-04-04 19:13:53 +03:00
|
|
|
switchMap(() => raf$), tap(() => elem.style[prop] = to),
|
2018-03-21 03:40:23 +02:00
|
|
|
switchMap(() => timer(getActualDuration(elem))), switchMap(() => this.void$),
|
|
|
|
);
|
2017-11-23 15:04:29 +02:00
|
|
|
};
|
|
|
|
|
2017-12-19 02:26:12 +02:00
|
|
|
const animateLeave = (elem: HTMLElement) => animateProp(elem, 'opacity', '1', '0.1');
|
|
|
|
const animateEnter = (elem: HTMLElement) => animateProp(elem, 'opacity', '0.1', '1');
|
2017-11-23 15:03:56 +02:00
|
|
|
|
|
|
|
let done$ = this.void$;
|
|
|
|
|
|
|
|
if (this.currViewContainer.parentElement) {
|
2018-03-21 03:40:23 +02:00
|
|
|
done$ = done$.pipe(
|
2017-11-23 15:03:56 +02:00
|
|
|
// Remove the current view from the viewer.
|
2018-03-21 03:40:23 +02:00
|
|
|
switchMap(() => animateLeave(this.currViewContainer)),
|
2020-10-16 21:46:24 +03:00
|
|
|
tap(() => (this.currViewContainer.parentElement as HTMLElement).removeChild(this.currViewContainer)),
|
2018-03-21 03:40:23 +02:00
|
|
|
tap(() => this.docRemoved.emit()),
|
|
|
|
);
|
2017-11-23 15:03:56 +02:00
|
|
|
}
|
|
|
|
|
2018-03-21 03:40:23 +02:00
|
|
|
return done$.pipe(
|
2017-11-23 15:03:56 +02:00
|
|
|
// Insert the next view into the viewer.
|
2018-03-21 03:40:23 +02:00
|
|
|
tap(() => this.hostElement.appendChild(this.nextViewContainer)),
|
|
|
|
tap(() => onInsertedCb()),
|
|
|
|
tap(() => this.docInserted.emit()),
|
|
|
|
switchMap(() => animateEnter(this.nextViewContainer)),
|
2017-11-23 15:03:56 +02:00
|
|
|
// Update the view references and clean up unused nodes.
|
2018-03-21 03:40:23 +02:00
|
|
|
tap(() => {
|
2017-11-23 15:03:56 +02:00
|
|
|
const prevViewContainer = this.currViewContainer;
|
|
|
|
this.currViewContainer = this.nextViewContainer;
|
|
|
|
this.nextViewContainer = prevViewContainer;
|
|
|
|
this.nextViewContainer.innerHTML = ''; // Empty to release memory.
|
2018-03-21 03:40:23 +02:00
|
|
|
}),
|
|
|
|
);
|
2017-11-23 15:03:56 +02:00
|
|
|
}
|
2017-03-01 14:30:25 +00:00
|
|
|
}
|