2016-06-23 09:47:54 -07:00
|
|
|
/**
|
|
|
|
* @license
|
|
|
|
* Copyright Google Inc. All Rights Reserved.
|
|
|
|
*
|
|
|
|
* Use of this source code is governed by an MIT-style license that can be
|
|
|
|
* found in the LICENSE file at https://angular.io/license
|
|
|
|
*/
|
|
|
|
|
2016-08-25 00:50:16 -07:00
|
|
|
import {SchemaMetadata} from '@angular/core';
|
2016-08-10 15:55:18 -07:00
|
|
|
|
2016-08-12 14:45:36 -07:00
|
|
|
import {CompileDirectiveMetadata, CompileIdentifierMetadata, CompileNgModuleMetadata, CompilePipeMetadata, CompileProviderMetadata, CompileTokenMetadata, StaticSymbol, createHostComponentMeta} from './compile_metadata';
|
2016-01-06 14:13:44 -08:00
|
|
|
import {DirectiveNormalizer} from './directive_normalizer';
|
2016-08-24 17:39:49 -07:00
|
|
|
import {Identifiers, resolveIdentifier, resolveIdentifierToken} from './identifiers';
|
2016-06-28 09:54:42 -07:00
|
|
|
import {CompileMetadataResolver} from './metadata_resolver';
|
2016-07-18 03:50:31 -07:00
|
|
|
import {NgModuleCompiler} from './ng_module_compiler';
|
2016-01-06 14:13:44 -08:00
|
|
|
import {OutputEmitter} from './output/abstract_emitter';
|
|
|
|
import * as o from './output/output_ast';
|
2016-06-24 08:46:43 -07:00
|
|
|
import {CompiledStylesheet, StyleCompiler} from './style_compiler';
|
2016-07-21 11:41:25 -07:00
|
|
|
import {TemplateParser} from './template_parser/template_parser';
|
2016-06-22 14:06:23 -07:00
|
|
|
import {ComponentFactoryDependency, ViewCompileResult, ViewCompiler, ViewFactoryDependency} from './view_compiler/view_compiler';
|
2016-01-06 14:13:44 -08:00
|
|
|
|
|
|
|
export class SourceModule {
|
|
|
|
constructor(public moduleUrl: string, public source: string) {}
|
|
|
|
}
|
|
|
|
|
2016-07-18 03:50:31 -07:00
|
|
|
export class NgModulesSummary {
|
|
|
|
constructor(public ngModuleByComponent: Map<StaticSymbol, CompileNgModuleMetadata>) {}
|
2016-06-28 09:54:42 -07:00
|
|
|
}
|
2016-07-18 03:50:31 -07:00
|
|
|
|
2016-01-06 14:13:44 -08:00
|
|
|
export class OfflineCompiler {
|
2016-06-08 16:38:52 -07:00
|
|
|
constructor(
|
2016-06-28 09:54:42 -07:00
|
|
|
private _metadataResolver: CompileMetadataResolver,
|
2016-06-08 16:38:52 -07:00
|
|
|
private _directiveNormalizer: DirectiveNormalizer, private _templateParser: TemplateParser,
|
|
|
|
private _styleCompiler: StyleCompiler, private _viewCompiler: ViewCompiler,
|
2016-08-12 14:45:36 -07:00
|
|
|
private _ngModuleCompiler: NgModuleCompiler, private _outputEmitter: OutputEmitter,
|
|
|
|
private _localeId: string, private _translationFormat: string) {}
|
2016-07-18 03:50:31 -07:00
|
|
|
|
|
|
|
analyzeModules(ngModules: StaticSymbol[]): NgModulesSummary {
|
|
|
|
const ngModuleByComponent = new Map<StaticSymbol, CompileNgModuleMetadata>();
|
|
|
|
|
|
|
|
ngModules.forEach((ngModule) => {
|
|
|
|
const ngModuleMeta = this._metadataResolver.getNgModuleMetadata(<any>ngModule);
|
|
|
|
ngModuleMeta.declaredDirectives.forEach((dirMeta) => {
|
|
|
|
if (dirMeta.isComponent) {
|
2016-08-29 08:52:25 -07:00
|
|
|
ngModuleByComponent.set(dirMeta.type.reference, ngModuleMeta);
|
2016-07-18 03:50:31 -07:00
|
|
|
}
|
|
|
|
});
|
2016-06-28 09:54:42 -07:00
|
|
|
});
|
2016-07-18 03:50:31 -07:00
|
|
|
return new NgModulesSummary(ngModuleByComponent);
|
2016-06-28 09:54:42 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
clearCache() {
|
|
|
|
this._directiveNormalizer.clearCache();
|
|
|
|
this._metadataResolver.clearCache();
|
|
|
|
}
|
|
|
|
|
|
|
|
compile(
|
2016-07-18 03:50:31 -07:00
|
|
|
moduleUrl: string, ngModulesSummary: NgModulesSummary, components: StaticSymbol[],
|
|
|
|
ngModules: StaticSymbol[]): Promise<SourceModule[]> {
|
2016-09-15 15:38:17 -07:00
|
|
|
const fileSuffix = _splitTypescriptSuffix(moduleUrl)[1];
|
|
|
|
const statements: o.Statement[] = [];
|
|
|
|
const exportedVars: string[] = [];
|
|
|
|
const outputSourceModules: SourceModule[] = [];
|
2016-06-28 09:54:42 -07:00
|
|
|
|
2016-07-18 03:50:31 -07:00
|
|
|
// compile all ng modules
|
2016-06-28 09:54:42 -07:00
|
|
|
exportedVars.push(
|
2016-07-18 03:50:31 -07:00
|
|
|
...ngModules.map((ngModuleType) => this._compileModule(ngModuleType, statements)));
|
2016-06-28 09:54:42 -07:00
|
|
|
|
|
|
|
// compile components
|
|
|
|
return Promise
|
|
|
|
.all(components.map((compType) => {
|
2016-07-18 03:50:31 -07:00
|
|
|
const compMeta = this._metadataResolver.getDirectiveMetadata(<any>compType);
|
2016-07-21 13:56:58 -07:00
|
|
|
const ngModule = ngModulesSummary.ngModuleByComponent.get(compType);
|
2016-07-18 03:50:31 -07:00
|
|
|
if (!ngModule) {
|
2016-08-25 00:50:16 -07:00
|
|
|
throw new Error(`Cannot determine the module for component ${compMeta.type.name}!`);
|
2016-06-28 09:54:42 -07:00
|
|
|
}
|
2016-09-15 15:38:17 -07:00
|
|
|
|
2016-06-28 09:54:42 -07:00
|
|
|
return Promise
|
2016-07-18 03:50:31 -07:00
|
|
|
.all([compMeta, ...ngModule.transitiveModule.directives].map(
|
|
|
|
dirMeta => this._directiveNormalizer.normalizeDirective(dirMeta).asyncResult))
|
2016-06-28 09:54:42 -07:00
|
|
|
.then((normalizedCompWithDirectives) => {
|
2016-09-15 15:38:17 -07:00
|
|
|
const [compMeta, ...dirMetas] = normalizedCompWithDirectives;
|
2016-06-28 09:54:42 -07:00
|
|
|
_assertComponent(compMeta);
|
|
|
|
|
|
|
|
// compile styles
|
2016-07-18 03:50:31 -07:00
|
|
|
const stylesCompileResults = this._styleCompiler.compileComponent(compMeta);
|
2016-06-28 09:54:42 -07:00
|
|
|
stylesCompileResults.externalStylesheets.forEach((compiledStyleSheet) => {
|
|
|
|
outputSourceModules.push(this._codgenStyles(compiledStyleSheet, fileSuffix));
|
|
|
|
});
|
|
|
|
|
|
|
|
// compile components
|
2016-09-15 15:38:17 -07:00
|
|
|
exportedVars.push(
|
|
|
|
this._compileComponentFactory(compMeta, fileSuffix, statements),
|
|
|
|
this._compileComponent(
|
2016-07-25 03:02:57 -07:00
|
|
|
compMeta, dirMetas, ngModule.transitiveModule.pipes, ngModule.schemas,
|
2016-07-18 03:50:31 -07:00
|
|
|
stylesCompileResults.componentStylesheet, fileSuffix, statements));
|
2016-06-28 09:54:42 -07:00
|
|
|
});
|
|
|
|
}))
|
|
|
|
.then(() => {
|
|
|
|
if (statements.length > 0) {
|
|
|
|
outputSourceModules.unshift(this._codegenSourceModule(
|
|
|
|
_ngfactoryModuleUrl(moduleUrl), statements, exportedVars));
|
|
|
|
}
|
|
|
|
return outputSourceModules;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2016-07-18 03:50:31 -07:00
|
|
|
private _compileModule(ngModuleType: StaticSymbol, targetStatements: o.Statement[]): string {
|
2016-09-15 15:38:17 -07:00
|
|
|
const ngModule = this._metadataResolver.getNgModuleMetadata(ngModuleType);
|
|
|
|
const appCompileResult = this._ngModuleCompiler.compile(ngModule, [
|
2016-08-24 17:39:49 -07:00
|
|
|
new CompileProviderMetadata(
|
|
|
|
{token: resolveIdentifierToken(Identifiers.LOCALE_ID), useValue: this._localeId}),
|
2016-08-12 14:45:36 -07:00
|
|
|
new CompileProviderMetadata({
|
2016-08-24 17:39:49 -07:00
|
|
|
token: resolveIdentifierToken(Identifiers.TRANSLATIONS_FORMAT),
|
2016-08-12 14:45:36 -07:00
|
|
|
useValue: this._translationFormat
|
|
|
|
})
|
|
|
|
]);
|
2016-06-28 09:54:42 -07:00
|
|
|
appCompileResult.dependencies.forEach((dep) => {
|
|
|
|
dep.placeholder.name = _componentFactoryName(dep.comp);
|
|
|
|
dep.placeholder.moduleUrl = _ngfactoryModuleUrl(dep.comp.moduleUrl);
|
|
|
|
});
|
|
|
|
targetStatements.push(...appCompileResult.statements);
|
2016-07-18 03:50:31 -07:00
|
|
|
return appCompileResult.ngModuleFactoryVar;
|
2016-06-28 09:54:42 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
private _compileComponentFactory(
|
|
|
|
compMeta: CompileDirectiveMetadata, fileSuffix: string,
|
|
|
|
targetStatements: o.Statement[]): string {
|
2016-09-15 15:38:17 -07:00
|
|
|
const hostMeta = createHostComponentMeta(compMeta);
|
|
|
|
const hostViewFactoryVar =
|
2016-07-25 03:02:57 -07:00
|
|
|
this._compileComponent(hostMeta, [compMeta], [], [], null, fileSuffix, targetStatements);
|
2016-09-15 15:38:17 -07:00
|
|
|
const compFactoryVar = _componentFactoryName(compMeta.type);
|
2016-06-28 09:54:42 -07:00
|
|
|
targetStatements.push(
|
|
|
|
o.variable(compFactoryVar)
|
2016-08-24 17:39:49 -07:00
|
|
|
.set(o.importExpr(resolveIdentifier(Identifiers.ComponentFactory), [o.importType(
|
|
|
|
compMeta.type)])
|
2016-06-28 09:54:42 -07:00
|
|
|
.instantiate(
|
|
|
|
[
|
2016-09-15 15:38:17 -07:00
|
|
|
o.literal(compMeta.selector),
|
|
|
|
o.variable(hostViewFactoryVar),
|
|
|
|
o.importExpr(compMeta.type),
|
2016-06-28 09:54:42 -07:00
|
|
|
],
|
|
|
|
o.importType(
|
2016-08-24 17:39:49 -07:00
|
|
|
resolveIdentifier(Identifiers.ComponentFactory),
|
|
|
|
[o.importType(compMeta.type)], [o.TypeModifier.Const])))
|
2016-06-28 09:54:42 -07:00
|
|
|
.toDeclStmt(null, [o.StmtModifier.Final]));
|
|
|
|
return compFactoryVar;
|
2016-01-06 14:13:44 -08:00
|
|
|
}
|
|
|
|
|
2016-06-08 16:38:52 -07:00
|
|
|
private _compileComponent(
|
|
|
|
compMeta: CompileDirectiveMetadata, directives: CompileDirectiveMetadata[],
|
2016-07-25 03:02:57 -07:00
|
|
|
pipes: CompilePipeMetadata[], schemas: SchemaMetadata[], componentStyles: CompiledStylesheet,
|
|
|
|
fileSuffix: string, targetStatements: o.Statement[]): string {
|
2016-09-15 15:38:17 -07:00
|
|
|
const parsedTemplate = this._templateParser.parse(
|
2016-07-25 03:02:57 -07:00
|
|
|
compMeta, compMeta.template.template, directives, pipes, schemas, compMeta.type.name);
|
2016-09-15 15:38:17 -07:00
|
|
|
const stylesExpr = componentStyles ? o.variable(componentStyles.stylesVar) : o.literalArr([]);
|
|
|
|
const viewResult =
|
2016-06-24 08:46:43 -07:00
|
|
|
this._viewCompiler.compileComponent(compMeta, parsedTemplate, stylesExpr, pipes);
|
|
|
|
if (componentStyles) {
|
2016-09-15 15:38:17 -07:00
|
|
|
targetStatements.push(..._resolveStyleStatements(componentStyles, fileSuffix));
|
2016-06-24 08:46:43 -07:00
|
|
|
}
|
2016-09-15 15:38:17 -07:00
|
|
|
targetStatements.push(..._resolveViewStatements(viewResult));
|
2016-01-06 14:13:44 -08:00
|
|
|
return viewResult.viewFactoryVar;
|
|
|
|
}
|
|
|
|
|
2016-06-24 08:46:43 -07:00
|
|
|
private _codgenStyles(stylesCompileResult: CompiledStylesheet, fileSuffix: string): SourceModule {
|
|
|
|
_resolveStyleStatements(stylesCompileResult, fileSuffix);
|
2016-06-08 16:38:52 -07:00
|
|
|
return this._codegenSourceModule(
|
2016-06-24 08:46:43 -07:00
|
|
|
_stylesModuleUrl(
|
|
|
|
stylesCompileResult.meta.moduleUrl, stylesCompileResult.isShimmed, fileSuffix),
|
|
|
|
stylesCompileResult.statements, [stylesCompileResult.stylesVar]);
|
2016-05-02 09:38:46 -07:00
|
|
|
}
|
2016-01-06 14:13:44 -08:00
|
|
|
|
2016-06-08 16:38:52 -07:00
|
|
|
private _codegenSourceModule(
|
|
|
|
moduleUrl: string, statements: o.Statement[], exportedVars: string[]): SourceModule {
|
2016-01-06 14:13:44 -08:00
|
|
|
return new SourceModule(
|
|
|
|
moduleUrl, this._outputEmitter.emitStatements(moduleUrl, statements, exportedVars));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function _resolveViewStatements(compileResult: ViewCompileResult): o.Statement[] {
|
2016-06-22 14:06:23 -07:00
|
|
|
compileResult.dependencies.forEach((dep) => {
|
|
|
|
if (dep instanceof ViewFactoryDependency) {
|
2016-09-15 15:38:17 -07:00
|
|
|
dep.placeholder.moduleUrl = _ngfactoryModuleUrl(dep.comp.moduleUrl);
|
2016-06-22 14:06:23 -07:00
|
|
|
} else if (dep instanceof ComponentFactoryDependency) {
|
2016-09-15 15:38:17 -07:00
|
|
|
dep.placeholder.name = _componentFactoryName(dep.comp);
|
|
|
|
dep.placeholder.moduleUrl = _ngfactoryModuleUrl(dep.comp.moduleUrl);
|
2016-06-22 14:06:23 -07:00
|
|
|
}
|
|
|
|
});
|
2016-01-06 14:13:44 -08:00
|
|
|
return compileResult.statements;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-06-08 16:38:52 -07:00
|
|
|
function _resolveStyleStatements(
|
2016-06-24 08:46:43 -07:00
|
|
|
compileResult: CompiledStylesheet, fileSuffix: string): o.Statement[] {
|
2016-01-06 14:13:44 -08:00
|
|
|
compileResult.dependencies.forEach((dep) => {
|
2016-06-24 08:46:43 -07:00
|
|
|
dep.valuePlaceholder.moduleUrl = _stylesModuleUrl(dep.moduleUrl, dep.isShimmed, fileSuffix);
|
2016-01-06 14:13:44 -08:00
|
|
|
});
|
|
|
|
return compileResult.statements;
|
|
|
|
}
|
|
|
|
|
2016-06-28 09:54:42 -07:00
|
|
|
function _ngfactoryModuleUrl(compUrl: string): string {
|
2016-09-15 15:38:17 -07:00
|
|
|
const urlWithSuffix = _splitTypescriptSuffix(compUrl);
|
2016-05-02 09:38:46 -07:00
|
|
|
return `${urlWithSuffix[0]}.ngfactory${urlWithSuffix[1]}`;
|
2016-01-06 14:13:44 -08:00
|
|
|
}
|
|
|
|
|
2016-06-22 14:06:23 -07:00
|
|
|
function _componentFactoryName(comp: CompileIdentifierMetadata): string {
|
|
|
|
return `${comp.name}NgFactory`;
|
|
|
|
}
|
|
|
|
|
2016-05-02 09:38:46 -07:00
|
|
|
function _stylesModuleUrl(stylesheetUrl: string, shim: boolean, suffix: string): string {
|
|
|
|
return shim ? `${stylesheetUrl}.shim${suffix}` : `${stylesheetUrl}${suffix}`;
|
2016-01-06 14:13:44 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
function _assertComponent(meta: CompileDirectiveMetadata) {
|
|
|
|
if (!meta.isComponent) {
|
2016-08-25 00:50:16 -07:00
|
|
|
throw new Error(`Could not compile '${meta.type.name}' because it is not a component.`);
|
2016-01-06 14:13:44 -08:00
|
|
|
}
|
|
|
|
}
|
2016-05-02 09:38:46 -07:00
|
|
|
|
2016-08-18 10:11:06 -07:00
|
|
|
function _splitTypescriptSuffix(path: string): string[] {
|
2016-09-15 15:38:17 -07:00
|
|
|
if (path.endsWith('.d.ts')) {
|
|
|
|
return [path.slice(0, -5), '.ts'];
|
2016-08-18 10:11:06 -07:00
|
|
|
}
|
2016-09-15 15:38:17 -07:00
|
|
|
|
|
|
|
const lastDot = path.lastIndexOf('.');
|
|
|
|
|
2016-05-02 09:38:46 -07:00
|
|
|
if (lastDot !== -1) {
|
|
|
|
return [path.substring(0, lastDot), path.substring(lastDot)];
|
|
|
|
}
|
2016-09-15 15:38:17 -07:00
|
|
|
|
|
|
|
return [path, ''];
|
2016-05-25 12:46:22 -07:00
|
|
|
}
|