-
Notifications
You must be signed in to change notification settings - Fork 39
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
1 parent
b8f4450
commit ef81da5
Showing
16 changed files
with
933 additions
and
7 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
7 changes: 7 additions & 0 deletions
7
packages/docs/src/app/containers/component-viwer/component-overview.template.html
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1 +1,8 @@ | ||
Overview for Component | ||
{{componentViewer.componentDocItem.packageName}}/{{componentViewer.componentDocItem.id}} | ||
<doc-viewer | ||
documentUrl="/docs-content/overviews/{{componentViewer.componentDocItem.packageName}}/{{componentViewer.componentDocItem.id}}.html" | ||
> | ||
|
||
</doc-viewer> | ||
<table-of-contents #toc container=".mat-drawer-content"></table-of-contents> |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
26 changes: 26 additions & 0 deletions
26
packages/docs/src/app/shared/doc-viewer/doc-viewer-module.ts
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,26 @@ | ||
import { CommonModule } from '@angular/common'; | ||
import { NgModule } from '@angular/core'; | ||
import { PortalModule } from '@ptsecurity/cdk/portal'; | ||
import { McButtonModule } from '@ptsecurity/mosaic/button'; | ||
import { McTabsModule } from '@ptsecurity/mosaic/tabs'; | ||
|
||
import { CopierService } from '../copier/copier.service'; | ||
import { ExampleViewer } from '../example-viewer/example-viewer'; | ||
|
||
import { DocViewer } from './doc-viewer'; | ||
|
||
|
||
// ExampleViewer is included in the DocViewerModule because they have a circular dependency. | ||
@NgModule({ | ||
imports: [ | ||
McButtonModule, | ||
McTabsModule, | ||
CommonModule, | ||
PortalModule | ||
], | ||
providers: [CopierService], | ||
declarations: [DocViewer, ExampleViewer], | ||
entryComponents: [ExampleViewer], | ||
exports: [DocViewer, ExampleViewer] | ||
}) | ||
export class DocViewerModule { } |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,130 @@ | ||
import { HttpClient, HttpErrorResponse } from '@angular/common/http'; | ||
import { | ||
ApplicationRef, | ||
Component, | ||
ComponentFactoryResolver, | ||
ElementRef, | ||
EventEmitter, | ||
Injector, | ||
Input, | ||
NgZone, | ||
OnDestroy, | ||
Output, | ||
SecurityContext, | ||
ViewContainerRef | ||
} from '@angular/core'; | ||
import { DomSanitizer } from '@angular/platform-browser'; | ||
import { ComponentPortal, DomPortalHost } from '@ptsecurity/cdk/portal'; | ||
import { Subscription } from 'rxjs'; | ||
import { take } from 'rxjs/operators'; | ||
|
||
import { ExampleViewer } from '../example-viewer/example-viewer'; | ||
|
||
|
||
@Component({ | ||
selector: 'doc-viewer', | ||
template: 'Loading document...' | ||
}) | ||
export class DocViewer implements OnDestroy { | ||
|
||
/** The URL of the document to display. */ | ||
@Input() | ||
set documentUrl(url: string) { | ||
this._fetchDocument(url); | ||
} | ||
|
||
@Output() contentRendered = new EventEmitter<void>(); | ||
|
||
/** The document text. It should not be HTML encoded. */ | ||
textContent = ''; | ||
private _portalHosts: DomPortalHost[] = []; | ||
private _documentFetchSubscription: Subscription; | ||
|
||
constructor(private _appRef: ApplicationRef, | ||
private _componentFactoryResolver: ComponentFactoryResolver, | ||
private _elementRef: ElementRef, | ||
private _http: HttpClient, | ||
private _injector: Injector, | ||
private _viewContainerRef: ViewContainerRef, | ||
private _ngZone: NgZone, | ||
private _domSanitizer: DomSanitizer) { | ||
} | ||
|
||
ngOnDestroy() { | ||
this._clearLiveExamples(); | ||
|
||
if (this._documentFetchSubscription) { | ||
this._documentFetchSubscription.unsubscribe(); | ||
} | ||
} | ||
|
||
/** Fetch a document by URL. */ | ||
private _fetchDocument(url: string) { | ||
// Cancel previous pending request | ||
if (this._documentFetchSubscription) { | ||
this._documentFetchSubscription.unsubscribe(); | ||
} | ||
|
||
this._documentFetchSubscription = this._http.get(url, {responseType: 'text'}).subscribe( | ||
(document) => this.updateDocument(document), | ||
(error) => this.showError(url, error) | ||
); | ||
} | ||
|
||
/** | ||
* Updates the displayed document. | ||
* @param rawDocument The raw document content to show. | ||
*/ | ||
private updateDocument(rawDocument: string) { | ||
// Replace all relative fragment URLs with absolute fragment URLs. e.g. "#my-section" becomes | ||
// "/components/button/api#my-section". This is necessary because otherwise these fragment | ||
// links would redirect to "/#my-section". | ||
rawDocument = rawDocument.replace(/href="#([^"]*)"/g, (_m: string, fragmentUrl: string) => { | ||
const absoluteUrl = `${location.pathname}#${fragmentUrl}`; | ||
|
||
return `href="${this._domSanitizer.sanitize(SecurityContext.URL, absoluteUrl)}"`; | ||
}); | ||
|
||
// tslint:disable-next-line:no-inner-html | ||
this._elementRef.nativeElement.innerHTML = rawDocument; | ||
this.textContent = this._elementRef.nativeElement.textContent; | ||
|
||
this._loadComponents('mosaic-docs-example', ExampleViewer); | ||
|
||
// Resolving and creating components dynamically in Angular happens synchronously, but since | ||
// we want to emit the output if the components are actually rendered completely, we wait | ||
// until the Angular zone becomes stable. | ||
this._ngZone.onStable | ||
.pipe(take(1)) | ||
.subscribe(() => this.contentRendered.next()); | ||
} | ||
|
||
/** Show an error that occurred when fetching a document. */ | ||
private showError(url: string, error: HttpErrorResponse) { | ||
console.log(error); | ||
this._elementRef.nativeElement.innerText = | ||
`Failed to load document: ${url}. Error: ${error.statusText}`; | ||
} | ||
|
||
/** Instantiate a ExampleViewer for each example. */ | ||
private _loadComponents(componentName: string, componentClass: any) { | ||
const exampleElements = | ||
this._elementRef.nativeElement.querySelectorAll(`[${componentName}]`); | ||
|
||
Array.prototype.slice.call(exampleElements).forEach((element: Element) => { | ||
const example = element.getAttribute(componentName); | ||
const portalHost = new DomPortalHost( | ||
element, this._componentFactoryResolver, this._appRef, this._injector); | ||
const examplePortal = new ComponentPortal(componentClass, this._viewContainerRef); | ||
const exampleViewer = portalHost.attach(examplePortal); | ||
(exampleViewer.instance as ExampleViewer).example = example; | ||
|
||
this._portalHosts.push(portalHost); | ||
}); | ||
} | ||
|
||
private _clearLiveExamples() { | ||
this._portalHosts.forEach((h) => h.dispose()); | ||
this._portalHosts = []; | ||
} | ||
} |
Oops, something went wrong.