172 lines
7.0 KiB
TypeScript
172 lines
7.0 KiB
TypeScript
/**
|
|
* @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
|
|
*/
|
|
|
|
/**
|
|
* Transform template html and css into executable code.
|
|
* Intended to be used in a build step.
|
|
*/
|
|
import * as compiler from '@angular/compiler';
|
|
import {ViewEncapsulation, lockRunMode} from '@angular/core';
|
|
import {AngularCompilerOptions} from '@angular/tsc-wrapped';
|
|
import * as path from 'path';
|
|
import * as ts from 'typescript';
|
|
|
|
import {CompileMetadataResolver, DirectiveNormalizer, DomElementSchemaRegistry, HtmlParser, Lexer, Parser, StyleCompiler, TemplateParser, TypeScriptEmitter, ViewCompiler} from './compiler_private';
|
|
import {ReflectorHost, ReflectorHostContext} from './reflector_host';
|
|
import {StaticAndDynamicReflectionCapabilities} from './static_reflection_capabilities';
|
|
import {StaticReflector} from './static_reflector';
|
|
|
|
const GENERATED_FILES = /\.ngfactory\.ts$|\.css\.ts$|\.css\.shim\.ts$/;
|
|
|
|
const PREAMBLE = `/**
|
|
* This file is generated by the Angular 2 template compiler.
|
|
* Do not edit.
|
|
*/
|
|
/* tslint:disable */
|
|
|
|
`;
|
|
|
|
export class CodeGenerator {
|
|
constructor(
|
|
private options: AngularCompilerOptions, private program: ts.Program,
|
|
public host: ts.CompilerHost, private staticReflector: StaticReflector,
|
|
private resolver: CompileMetadataResolver, private compiler: compiler.OfflineCompiler,
|
|
private reflectorHost: ReflectorHost) {
|
|
lockRunMode();
|
|
}
|
|
|
|
private generateSource(metadatas: compiler.CompileDirectiveMetadata[]) {
|
|
const normalize = (metadata: compiler.CompileDirectiveMetadata) => {
|
|
const directiveType = metadata.type.runtime;
|
|
const directives = this.resolver.getViewDirectivesMetadata(directiveType);
|
|
return Promise.all(directives.map(d => this.compiler.normalizeDirectiveMetadata(d)))
|
|
.then(normalizedDirectives => {
|
|
const pipes = this.resolver.getViewPipesMetadata(directiveType);
|
|
return new compiler.NormalizedComponentWithViewDirectives(
|
|
metadata, normalizedDirectives, pipes);
|
|
});
|
|
};
|
|
return Promise.all(metadatas.map(normalize))
|
|
.then(
|
|
normalizedCompWithDirectives =>
|
|
this.compiler.compileTemplates(normalizedCompWithDirectives));
|
|
}
|
|
|
|
private readComponents(absSourcePath: string) {
|
|
const result: Promise<compiler.CompileDirectiveMetadata>[] = [];
|
|
const moduleMetadata = this.staticReflector.getModuleMetadata(absSourcePath);
|
|
if (!moduleMetadata) {
|
|
console.log(`WARNING: no metadata found for ${absSourcePath}`);
|
|
return result;
|
|
}
|
|
const metadata = moduleMetadata['metadata'];
|
|
const symbols = metadata && Object.keys(metadata);
|
|
if (!symbols || !symbols.length) {
|
|
return result;
|
|
}
|
|
for (const symbol of symbols) {
|
|
if (metadata[symbol] && metadata[symbol].__symbolic == 'error') {
|
|
// Ignore symbols that are only included to record error information.
|
|
continue;
|
|
}
|
|
const staticType = this.reflectorHost.findDeclaration(absSourcePath, symbol, absSourcePath);
|
|
let directive: compiler.CompileDirectiveMetadata;
|
|
directive = this.resolver.maybeGetDirectiveMetadata(<any>staticType);
|
|
|
|
if (!directive || !directive.isComponent) {
|
|
continue;
|
|
}
|
|
result.push(this.compiler.normalizeDirectiveMetadata(directive));
|
|
}
|
|
return result;
|
|
}
|
|
|
|
// Write codegen in a directory structure matching the sources.
|
|
private calculateEmitPath(filePath: string) {
|
|
let root = this.options.basePath;
|
|
for (let eachRootDir of this.options.rootDirs || []) {
|
|
if (this.options.trace) {
|
|
console.log(`Check if ${filePath} is under rootDirs element ${eachRootDir}`);
|
|
}
|
|
if (path.relative(eachRootDir, filePath).indexOf('.') !== 0) {
|
|
root = eachRootDir;
|
|
}
|
|
}
|
|
|
|
return path.join(this.options.genDir, path.relative(root, filePath));
|
|
}
|
|
|
|
codegen(): Promise<any> {
|
|
const generateOneFile = (absSourcePath: string) =>
|
|
Promise.all(this.readComponents(absSourcePath))
|
|
.then((metadatas: compiler.CompileDirectiveMetadata[]) => {
|
|
if (!metadatas || !metadatas.length) {
|
|
return;
|
|
}
|
|
return this.generateSource(metadatas);
|
|
})
|
|
.then(generatedModules => {
|
|
if (generatedModules) {
|
|
generatedModules.forEach((generatedModule) => {
|
|
const sourceFile = this.program.getSourceFile(absSourcePath);
|
|
const emitPath = this.calculateEmitPath(generatedModule.moduleUrl);
|
|
this.host.writeFile(
|
|
emitPath, PREAMBLE + generatedModule.source, false, () => {}, [sourceFile]);
|
|
});
|
|
}
|
|
})
|
|
.catch((e) => { console.error(e.stack); });
|
|
var compPromises = this.program.getSourceFiles()
|
|
.map(sf => sf.fileName)
|
|
.filter(f => !GENERATED_FILES.test(f))
|
|
.map(generateOneFile);
|
|
return Promise.all(compPromises);
|
|
}
|
|
|
|
static create(
|
|
options: AngularCompilerOptions, program: ts.Program, compilerHost: ts.CompilerHost,
|
|
reflectorHostContext?: ReflectorHostContext): CodeGenerator {
|
|
const xhr: compiler.XHR = {
|
|
get: (s: string) => {
|
|
if (!compilerHost.fileExists(s)) {
|
|
// TODO: We should really have a test for error cases like this!
|
|
throw new Error(`Compilation failed. Resource file not found: ${s}`);
|
|
}
|
|
return Promise.resolve(compilerHost.readFile(s));
|
|
}
|
|
};
|
|
const urlResolver: compiler.UrlResolver = compiler.createOfflineCompileUrlResolver();
|
|
const reflectorHost = new ReflectorHost(program, compilerHost, options, reflectorHostContext);
|
|
const staticReflector = new StaticReflector(reflectorHost);
|
|
StaticAndDynamicReflectionCapabilities.install(staticReflector);
|
|
const htmlParser = new HtmlParser();
|
|
const config = new compiler.CompilerConfig({
|
|
genDebugInfo: options.debug === true,
|
|
defaultEncapsulation: ViewEncapsulation.Emulated,
|
|
logBindingUpdate: false,
|
|
useJit: false,
|
|
platformDirectives: [],
|
|
platformPipes: []
|
|
});
|
|
const normalizer = new DirectiveNormalizer(xhr, urlResolver, htmlParser, config);
|
|
const parser = new Parser(new Lexer());
|
|
const tmplParser = new TemplateParser(
|
|
parser, new DomElementSchemaRegistry(), htmlParser,
|
|
/*console*/ null, []);
|
|
const offlineCompiler = new compiler.OfflineCompiler(
|
|
normalizer, tmplParser, new StyleCompiler(urlResolver), new ViewCompiler(config),
|
|
new TypeScriptEmitter(reflectorHost));
|
|
const resolver = new CompileMetadataResolver(
|
|
new compiler.DirectiveResolver(staticReflector), new compiler.PipeResolver(staticReflector),
|
|
new compiler.ViewResolver(staticReflector), config, staticReflector);
|
|
|
|
return new CodeGenerator(
|
|
options, program, compilerHost, staticReflector, resolver, offlineCompiler, reflectorHost);
|
|
}
|
|
}
|