| 
									
										
										
										
											2017-03-01 14:30:25 +00:00
										 |  |  | import { | 
					
						
							|  |  |  |   Component, ComponentFactory, ComponentFactoryResolver, ComponentRef, | 
					
						
							| 
									
										
										
										
											2017-03-13 09:20:09 +00:00
										 |  |  |   DoCheck, ElementRef, EventEmitter, Injector, Input, OnDestroy, | 
					
						
							|  |  |  |   Output, ViewEncapsulation | 
					
						
							| 
									
										
										
										
											2017-03-01 14:30:25 +00:00
										 |  |  | } from '@angular/core'; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-26 13:32:29 -07:00
										 |  |  | import { EmbeddedComponents } from 'app/embedded/embedded.module'; | 
					
						
							| 
									
										
										
										
											2017-03-01 14:30:25 +00:00
										 |  |  | import { DocumentContents } from 'app/documents/document.service'; | 
					
						
							| 
									
										
										
										
											2017-04-27 15:32:46 -07:00
										 |  |  | import { Title } from '@angular/platform-browser'; | 
					
						
							|  |  |  | import { TocService } from 'app/shared/toc.service'; | 
					
						
							| 
									
										
										
										
											2017-03-01 14:30:25 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | interface EmbeddedComponentFactory { | 
					
						
							|  |  |  |   contentPropertyName: string; | 
					
						
							|  |  |  |   factory: ComponentFactory<any>; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // 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 (?!)
 | 
					
						
							|  |  |  |   // encapsulation: ViewEncapsulation.Native
 | 
					
						
							|  |  |  | }) | 
					
						
							|  |  |  | export class DocViewerComponent implements DoCheck, OnDestroy { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-04 00:37:55 -07:00
										 |  |  |   private embeddedComponents: ComponentRef<any>[] = []; | 
					
						
							| 
									
										
										
										
											2017-03-01 14:30:25 +00:00
										 |  |  |   private embeddedComponentFactories: Map<string, EmbeddedComponentFactory> = new Map(); | 
					
						
							|  |  |  |   private hostElement: HTMLElement; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-13 09:20:09 +00:00
										 |  |  |   @Output() | 
					
						
							| 
									
										
										
										
											2017-04-01 21:45:32 -07:00
										 |  |  |   docRendered = new EventEmitter(); | 
					
						
							| 
									
										
										
										
											2017-03-13 09:20:09 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-01 14:30:25 +00:00
										 |  |  |   constructor( | 
					
						
							|  |  |  |     componentFactoryResolver: ComponentFactoryResolver, | 
					
						
							|  |  |  |     elementRef: ElementRef, | 
					
						
							|  |  |  |     embeddedComponents: EmbeddedComponents, | 
					
						
							| 
									
										
										
										
											2017-04-27 15:32:46 -07:00
										 |  |  |     private injector: Injector, | 
					
						
							|  |  |  |     private titleService: Title, | 
					
						
							|  |  |  |     private tocService: TocService | 
					
						
							| 
									
										
										
										
											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; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (const component of embeddedComponents.components) { | 
					
						
							|  |  |  |       const factory = componentFactoryResolver.resolveComponentFactory(component); | 
					
						
							|  |  |  |       const selector = factory.selector; | 
					
						
							|  |  |  |       const contentPropertyName = this.selectorToContentPropertyName(selector); | 
					
						
							|  |  |  |       this.embeddedComponentFactories.set(selector, { contentPropertyName, factory }); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   @Input() | 
					
						
							|  |  |  |   set doc(newDoc: DocumentContents) { | 
					
						
							|  |  |  |     this.ngOnDestroy(); | 
					
						
							|  |  |  |     if (newDoc) { | 
					
						
							|  |  |  |       this.build(newDoc); | 
					
						
							| 
									
										
										
										
											2017-04-01 21:45:32 -07:00
										 |  |  |       this.docRendered.emit(); | 
					
						
							| 
									
										
										
										
											2017-03-01 14:30:25 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /** | 
					
						
							|  |  |  |    * Add doc content to host element and build it out with embedded components | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   private build(doc: DocumentContents) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // security: the doc.content is always authored by the documentation team
 | 
					
						
							|  |  |  |     // and is considered to be safe
 | 
					
						
							|  |  |  |     this.hostElement.innerHTML = doc.contents || ''; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!doc.contents) { return; } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-27 15:32:46 -07:00
										 |  |  |     this.addTitleAndToc(doc.id); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-01 14:30:25 +00:00
										 |  |  |     // TODO(i): why can't I use for-of? why doesn't typescript like Map#value() iterators?
 | 
					
						
							|  |  |  |     this.embeddedComponentFactories.forEach(({ contentPropertyName, factory }, selector) => { | 
					
						
							|  |  |  |       const embeddedComponentElements = this.hostElement.querySelectorAll(selector); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       // cast due to https://github.com/Microsoft/TypeScript/issues/4947
 | 
					
						
							|  |  |  |       for (const element of embeddedComponentElements as any as HTMLElement[]){ | 
					
						
							|  |  |  |         // hack: preserve the current element content because the factory will empty it out
 | 
					
						
							|  |  |  |         // security: the source of this innerHTML is always authored by the documentation team
 | 
					
						
							|  |  |  |         // and is considered to be safe
 | 
					
						
							|  |  |  |         element[contentPropertyName] = element.innerHTML; | 
					
						
							| 
									
										
										
										
											2017-05-04 00:37:55 -07:00
										 |  |  |         this.embeddedComponents.push(factory.create(this.injector, [], element)); | 
					
						
							| 
									
										
										
										
											2017-03-01 14:30:25 +00:00
										 |  |  |       } | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-27 15:32:46 -07:00
										 |  |  |   private addTitleAndToc(docId: string) { | 
					
						
							|  |  |  |     this.tocService.reset(); | 
					
						
							|  |  |  |     let title = ''; | 
					
						
							|  |  |  |     const titleEl = this.hostElement.querySelector('h1'); | 
					
						
							|  |  |  |     // Only create TOC for docs with an <h1> title
 | 
					
						
							| 
									
										
										
										
											2017-04-27 15:33:50 -07:00
										 |  |  |     // If you don't want a TOC, add "no-toc" class to <h1>
 | 
					
						
							| 
									
										
										
										
											2017-04-27 15:32:46 -07:00
										 |  |  |     if (titleEl) { | 
					
						
							|  |  |  |       title = titleEl.innerText.trim(); | 
					
						
							|  |  |  |       if (!/(no-toc|notoc)/i.test(titleEl.className)) { | 
					
						
							|  |  |  |         this.tocService.genToc(this.hostElement, docId); | 
					
						
							|  |  |  |         titleEl.insertAdjacentHTML('afterend', '<aio-toc class="embedded"></aio-toc>'); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     this.titleService.setTitle(title ? `Angular - ${title}` : 'Angular'); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-01 14:30:25 +00:00
										 |  |  |   ngDoCheck() { | 
					
						
							| 
									
										
										
										
											2017-05-04 00:37:55 -07:00
										 |  |  |     this.embeddedComponents.forEach(comp => comp.changeDetectorRef.detectChanges()); | 
					
						
							| 
									
										
										
										
											2017-03-01 14:30:25 +00:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   ngOnDestroy() { | 
					
						
							| 
									
										
										
										
											2017-05-04 00:37:55 -07:00
										 |  |  |     // destroy these components else there will be memory leaks
 | 
					
						
							|  |  |  |     this.embeddedComponents.forEach(comp => comp.destroy()); | 
					
						
							|  |  |  |     this.embeddedComponents.length = 0; | 
					
						
							| 
									
										
										
										
											2017-03-01 14:30:25 +00:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /** | 
					
						
							|  |  |  |    * Compute the component content property name by converting the selector to camelCase and appending | 
					
						
							|  |  |  |    * 'Content', e.g. live-example => liveExampleContent | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   private selectorToContentPropertyName(selector: string) { | 
					
						
							|  |  |  |     return selector.replace(/-(.)/g, (match, $1) => $1.toUpperCase()) + 'Content'; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } |