2018-10-24 19:02:25 -04: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
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
import {CompilerFacade, CoreEnvironment, ExportedCompilerFacade, R3ComponentMetadataFacade, R3DependencyMetadataFacade, R3DirectiveMetadataFacade, R3InjectableMetadataFacade, R3InjectorMetadataFacade, R3NgModuleMetadataFacade, R3PipeMetadataFacade, R3QueryMetadataFacade, StringMap, StringMapWithRename} from './compiler_facade_interface';
|
|
|
|
import {ConstantPool} from './constant_pool';
|
|
|
|
import {HostBinding, HostListener, Input, Output, Type} from './core';
|
|
|
|
import {compileInjectable} from './injectable_compiler_2';
|
|
|
|
import {Expression, LiteralExpr, WrappedNodeExpr} from './output/output_ast';
|
|
|
|
import {R3DependencyMetadata, R3ResolvedDependencyType} from './render3/r3_factory';
|
|
|
|
import {jitExpression} from './render3/r3_jit';
|
|
|
|
import {R3InjectorMetadata, R3NgModuleMetadata, compileInjector, compileNgModule} from './render3/r3_module_compiler';
|
|
|
|
import {compilePipeFromMetadata} from './render3/r3_pipe_compiler';
|
|
|
|
import {R3Reference} from './render3/util';
|
|
|
|
import {R3DirectiveMetadata, R3QueryMetadata} from './render3/view/api';
|
|
|
|
import {compileComponentFromMetadata, compileDirectiveFromMetadata, parseHostBindings} from './render3/view/compiler';
|
|
|
|
import {makeBindingParser, parseTemplate} from './render3/view/template';
|
|
|
|
|
|
|
|
export class CompilerFacadeImpl implements CompilerFacade {
|
|
|
|
R3ResolvedDependencyType = R3ResolvedDependencyType as any;
|
|
|
|
|
|
|
|
compilePipe(angularCoreEnv: CoreEnvironment, sourceMapUrl: string, facade: R3PipeMetadataFacade):
|
|
|
|
any {
|
|
|
|
const res = compilePipeFromMetadata({
|
|
|
|
name: facade.name,
|
|
|
|
type: new WrappedNodeExpr(facade.type),
|
|
|
|
deps: convertR3DependencyMetadataArray(facade.deps),
|
|
|
|
pipeName: facade.pipeName,
|
|
|
|
pure: facade.pure,
|
|
|
|
});
|
|
|
|
return jitExpression(res.expression, angularCoreEnv, sourceMapUrl, res.statements);
|
|
|
|
}
|
|
|
|
|
|
|
|
compileInjectable(
|
|
|
|
angularCoreEnv: CoreEnvironment, sourceMapUrl: string,
|
|
|
|
facade: R3InjectableMetadataFacade): any {
|
|
|
|
const {expression, statements} = compileInjectable({
|
|
|
|
name: facade.name,
|
|
|
|
type: new WrappedNodeExpr(facade.type),
|
|
|
|
providedIn: computeProvidedIn(facade.providedIn),
|
|
|
|
useClass: wrapExpression(facade, USE_CLASS),
|
|
|
|
useFactory: wrapExpression(facade, USE_FACTORY),
|
|
|
|
useValue: wrapExpression(facade, USE_VALUE),
|
|
|
|
useExisting: wrapExpression(facade, USE_EXISTING),
|
|
|
|
ctorDeps: convertR3DependencyMetadataArray(facade.ctorDeps),
|
|
|
|
userDeps: convertR3DependencyMetadataArray(facade.userDeps) || undefined,
|
|
|
|
});
|
|
|
|
|
|
|
|
return jitExpression(expression, angularCoreEnv, sourceMapUrl, statements);
|
|
|
|
}
|
|
|
|
|
|
|
|
compileInjector(
|
|
|
|
angularCoreEnv: CoreEnvironment, sourceMapUrl: string,
|
|
|
|
facade: R3InjectorMetadataFacade): any {
|
|
|
|
const meta: R3InjectorMetadata = {
|
|
|
|
name: facade.name,
|
|
|
|
type: new WrappedNodeExpr(facade.type),
|
|
|
|
deps: convertR3DependencyMetadataArray(facade.deps),
|
|
|
|
providers: new WrappedNodeExpr(facade.providers),
|
|
|
|
imports: new WrappedNodeExpr(facade.imports),
|
|
|
|
};
|
|
|
|
const res = compileInjector(meta);
|
|
|
|
return jitExpression(res.expression, angularCoreEnv, sourceMapUrl, res.statements);
|
|
|
|
}
|
|
|
|
|
|
|
|
compileNgModule(
|
|
|
|
angularCoreEnv: CoreEnvironment, sourceMapUrl: string,
|
|
|
|
facade: R3NgModuleMetadataFacade): any {
|
|
|
|
const meta: R3NgModuleMetadata = {
|
|
|
|
type: new WrappedNodeExpr(facade.type),
|
|
|
|
bootstrap: facade.bootstrap.map(wrapReference),
|
|
|
|
declarations: facade.declarations.map(wrapReference),
|
|
|
|
imports: facade.imports.map(wrapReference),
|
|
|
|
exports: facade.exports.map(wrapReference),
|
|
|
|
emitInline: true,
|
|
|
|
};
|
|
|
|
const res = compileNgModule(meta);
|
|
|
|
return jitExpression(res.expression, angularCoreEnv, sourceMapUrl, []);
|
|
|
|
}
|
|
|
|
|
|
|
|
compileDirective(
|
|
|
|
angularCoreEnv: CoreEnvironment, sourceMapUrl: string,
|
|
|
|
facade: R3DirectiveMetadataFacade): any {
|
|
|
|
const constantPool = new ConstantPool();
|
|
|
|
const bindingParser = makeBindingParser();
|
|
|
|
|
|
|
|
const meta: R3DirectiveMetadata = convertDirectiveFacadeToMetadata(facade);
|
|
|
|
const res = compileDirectiveFromMetadata(meta, constantPool, bindingParser);
|
|
|
|
const preStatements = [...constantPool.statements, ...res.statements];
|
|
|
|
return jitExpression(res.expression, angularCoreEnv, sourceMapUrl, preStatements);
|
|
|
|
}
|
|
|
|
|
|
|
|
compileComponent(
|
|
|
|
angularCoreEnv: CoreEnvironment, sourceMapUrl: string,
|
|
|
|
facade: R3ComponentMetadataFacade): any {
|
|
|
|
// The ConstantPool is a requirement of the JIT'er.
|
|
|
|
const constantPool = new ConstantPool();
|
|
|
|
|
|
|
|
// Parse the template and check for errors.
|
|
|
|
const template = parseTemplate(
|
|
|
|
facade.template, sourceMapUrl, {
|
|
|
|
preserveWhitespaces: facade.preserveWhitespaces || false,
|
|
|
|
},
|
|
|
|
'');
|
|
|
|
if (template.errors !== undefined) {
|
|
|
|
const errors = template.errors.map(err => err.toString()).join(', ');
|
|
|
|
throw new Error(`Errors during JIT compilation of template for ${facade.name}: ${errors}`);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Compile the component metadata, including template, into an expression.
|
|
|
|
// TODO(alxhub): implement inputs, outputs, queries, etc.
|
|
|
|
const res = compileComponentFromMetadata(
|
|
|
|
{
|
|
|
|
...facade as R3ComponentMetadataFacadeNoPropAndWhitespace,
|
|
|
|
...convertDirectiveFacadeToMetadata(facade),
|
|
|
|
template,
|
|
|
|
viewQueries: facade.viewQueries.map(convertToR3QueryMetadata),
|
|
|
|
wrapDirectivesAndPipesInClosure: false,
|
|
|
|
styles: facade.styles || [],
|
|
|
|
encapsulation: facade.encapsulation as any,
|
|
|
|
animations: facade.animations != null ? new WrappedNodeExpr(facade.animations) : null,
|
|
|
|
viewProviders: facade.viewProviders != null ? new WrappedNodeExpr(facade.viewProviders) :
|
|
|
|
null,
|
|
|
|
},
|
|
|
|
constantPool, makeBindingParser());
|
|
|
|
const preStatements = [...constantPool.statements, ...res.statements];
|
|
|
|
|
|
|
|
return jitExpression(res.expression, angularCoreEnv, sourceMapUrl, preStatements);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// This seems to be needed to placate TS v3.0 only
|
|
|
|
type R3ComponentMetadataFacadeNoPropAndWhitespace = Pick<
|
|
|
|
R3ComponentMetadataFacade,
|
|
|
|
Exclude<Exclude<keyof R3ComponentMetadataFacade, 'preserveWhitespaces'>, 'propMetadata'>>;
|
|
|
|
|
|
|
|
const USE_CLASS = Object.keys({useClass: null})[0];
|
|
|
|
const USE_FACTORY = Object.keys({useFactory: null})[0];
|
|
|
|
const USE_VALUE = Object.keys({useValue: null})[0];
|
|
|
|
const USE_EXISTING = Object.keys({useExisting: null})[0];
|
|
|
|
|
|
|
|
const wrapReference = function(value: Type): R3Reference {
|
|
|
|
const wrapped = new WrappedNodeExpr(value);
|
|
|
|
return {value: wrapped, type: wrapped};
|
|
|
|
};
|
|
|
|
|
|
|
|
function convertToR3QueryMetadata(facade: R3QueryMetadataFacade): R3QueryMetadata {
|
|
|
|
return {
|
|
|
|
...facade,
|
|
|
|
predicate: Array.isArray(facade.predicate) ? facade.predicate :
|
|
|
|
new WrappedNodeExpr(facade.predicate),
|
|
|
|
read: facade.read ? new WrappedNodeExpr(facade.read) : null,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
function convertDirectiveFacadeToMetadata(facade: R3DirectiveMetadataFacade): R3DirectiveMetadata {
|
|
|
|
const inputsFromMetadata = parseInputOutputs(facade.inputs || []);
|
|
|
|
const outputsFromMetadata = parseInputOutputs(facade.outputs || []);
|
|
|
|
const propMetadata = facade.propMetadata;
|
|
|
|
const inputsFromType: StringMapWithRename = {};
|
|
|
|
const outputsFromType: StringMap = {};
|
|
|
|
for (const field in propMetadata) {
|
|
|
|
if (propMetadata.hasOwnProperty(field)) {
|
|
|
|
propMetadata[field].forEach(ann => {
|
|
|
|
if (isInput(ann)) {
|
|
|
|
inputsFromType[field] =
|
|
|
|
ann.bindingPropertyName ? [ann.bindingPropertyName, field] : field;
|
|
|
|
} else if (isOutput(ann)) {
|
|
|
|
outputsFromType[field] = ann.bindingPropertyName || field;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return {
|
|
|
|
...facade as R3DirectiveMetadataFacadeNoPropAndWhitespace,
|
|
|
|
typeSourceSpan: null !,
|
|
|
|
type: new WrappedNodeExpr(facade.type),
|
|
|
|
deps: convertR3DependencyMetadataArray(facade.deps),
|
|
|
|
host: extractHostBindings(facade.host, facade.propMetadata),
|
|
|
|
inputs: {...inputsFromMetadata, ...inputsFromType},
|
|
|
|
outputs: {...outputsFromMetadata, ...outputsFromType},
|
2018-11-12 21:02:47 -05:00
|
|
|
queries: facade.queries.map(convertToR3QueryMetadata),
|
2018-10-24 19:02:25 -04:00
|
|
|
providers: facade.providers != null ? new WrappedNodeExpr(facade.providers) : null,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
// This seems to be needed to placate TS v3.0 only
|
|
|
|
type R3DirectiveMetadataFacadeNoPropAndWhitespace =
|
|
|
|
Pick<R3DirectiveMetadataFacade, Exclude<keyof R3DirectiveMetadataFacade, 'propMetadata'>>;
|
|
|
|
|
|
|
|
function wrapExpression(obj: any, property: string): WrappedNodeExpr<any>|undefined {
|
|
|
|
if (obj.hasOwnProperty(property)) {
|
|
|
|
return new WrappedNodeExpr(obj[property]);
|
|
|
|
} else {
|
|
|
|
return undefined;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function computeProvidedIn(providedIn: Type | string | null | undefined): Expression {
|
|
|
|
if (providedIn == null || typeof providedIn === 'string') {
|
|
|
|
return new LiteralExpr(providedIn);
|
|
|
|
} else {
|
|
|
|
return new WrappedNodeExpr(providedIn);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function convertR3DependencyMetadata(facade: R3DependencyMetadataFacade): R3DependencyMetadata {
|
|
|
|
let tokenExpr;
|
|
|
|
if (facade.token === null) {
|
|
|
|
tokenExpr = new LiteralExpr(null);
|
|
|
|
} else if (facade.resolved === R3ResolvedDependencyType.Attribute) {
|
|
|
|
tokenExpr = new LiteralExpr(facade.token);
|
|
|
|
} else {
|
|
|
|
tokenExpr = new WrappedNodeExpr(facade.token);
|
|
|
|
}
|
|
|
|
return {
|
|
|
|
token: tokenExpr,
|
|
|
|
resolved: facade.resolved,
|
|
|
|
host: facade.host,
|
|
|
|
optional: facade.optional,
|
|
|
|
self: facade.self,
|
|
|
|
skipSelf: facade.skipSelf
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
function convertR3DependencyMetadataArray(facades: R3DependencyMetadataFacade[] | null | undefined):
|
|
|
|
R3DependencyMetadata[]|null {
|
|
|
|
return facades == null ? null : facades.map(convertR3DependencyMetadata);
|
|
|
|
}
|
|
|
|
|
|
|
|
function extractHostBindings(host: {[key: string]: string}, propMetadata: {[key: string]: any[]}): {
|
|
|
|
attributes: StringMap,
|
|
|
|
listeners: StringMap,
|
|
|
|
properties: StringMap,
|
|
|
|
} {
|
|
|
|
// First parse the declarations from the metadata.
|
|
|
|
const {attributes, listeners, properties, animations} = parseHostBindings(host || {});
|
|
|
|
|
|
|
|
if (Object.keys(animations).length > 0) {
|
|
|
|
throw new Error(`Animation bindings are as-of-yet unsupported in Ivy`);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Next, loop over the properties of the object, looking for @HostBinding and @HostListener.
|
|
|
|
for (const field in propMetadata) {
|
|
|
|
if (propMetadata.hasOwnProperty(field)) {
|
|
|
|
propMetadata[field].forEach(ann => {
|
|
|
|
if (isHostBinding(ann)) {
|
|
|
|
properties[ann.hostPropertyName || field] = field;
|
|
|
|
} else if (isHostListener(ann)) {
|
|
|
|
listeners[ann.eventName || field] = `${field}(${(ann.args || []).join(',')})`;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return {attributes, listeners, properties};
|
|
|
|
}
|
|
|
|
|
|
|
|
function isHostBinding(value: any): value is HostBinding {
|
|
|
|
return value.ngMetadataName === 'HostBinding';
|
|
|
|
}
|
|
|
|
|
|
|
|
function isHostListener(value: any): value is HostListener {
|
|
|
|
return value.ngMetadataName === 'HostListener';
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
function isInput(value: any): value is Input {
|
|
|
|
return value.ngMetadataName === 'Input';
|
|
|
|
}
|
|
|
|
|
|
|
|
function isOutput(value: any): value is Output {
|
|
|
|
return value.ngMetadataName === 'Output';
|
|
|
|
}
|
|
|
|
|
|
|
|
function parseInputOutputs(values: string[]): StringMap {
|
|
|
|
return values.reduce(
|
|
|
|
(map, value) => {
|
|
|
|
const [field, property] = value.split(',').map(piece => piece.trim());
|
|
|
|
map[field] = property || field;
|
|
|
|
return map;
|
|
|
|
},
|
|
|
|
{} as StringMap);
|
|
|
|
}
|
|
|
|
|
|
|
|
export function publishFacade(global: any) {
|
|
|
|
const ng: ExportedCompilerFacade = global.ng || (global.ng = {});
|
|
|
|
ng.ɵcompilerFacade = new CompilerFacadeImpl();
|
2018-11-12 21:02:47 -05:00
|
|
|
}
|