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
|
|
|
|
*/
|
|
|
|
|
2017-10-04 10:51:34 -07:00
|
|
|
import {CompileDirectiveMetadata, CompileStylesheetMetadata, CompileTemplateMetadata, templateSourceUrl} from './compile_metadata';
|
2017-07-28 15:58:28 +02:00
|
|
|
import {CompilerConfig, preserveWhitespacesDefault} from './config';
|
2017-08-16 09:00:03 -07:00
|
|
|
import {ViewEncapsulation} from './core';
|
2016-08-01 12:19:09 -07:00
|
|
|
import * as html from './ml_parser/ast';
|
|
|
|
import {HtmlParser} from './ml_parser/html_parser';
|
|
|
|
import {InterpolationConfig} from './ml_parser/interpolation_config';
|
2017-09-12 09:40:28 -07:00
|
|
|
import {ParseTreeResult as HtmlParseTreeResult} from './ml_parser/parser';
|
2016-08-17 09:24:44 -07:00
|
|
|
import {ResourceLoader} from './resource_loader';
|
2016-06-24 08:46:43 -07:00
|
|
|
import {extractStyleUrls, isStyleUrlResolvable} from './style_url_resolver';
|
2016-07-21 11:41:25 -07:00
|
|
|
import {PreparsedElementType, preparseElement} from './template_parser/template_preparser';
|
2016-06-24 08:46:43 -07:00
|
|
|
import {UrlResolver} from './url_resolver';
|
2017-08-16 09:00:03 -07:00
|
|
|
import {SyncAsync, isDefined, stringify, syntaxError} from './util';
|
2015-08-25 15:36:02 -07:00
|
|
|
|
2016-11-10 14:07:30 -08:00
|
|
|
export interface PrenormalizedTemplateMetadata {
|
2017-03-14 09:16:15 -07:00
|
|
|
ngModuleType: any;
|
2016-11-10 14:07:30 -08:00
|
|
|
componentType: any;
|
|
|
|
moduleUrl: string;
|
2017-03-24 09:59:58 -07:00
|
|
|
template: string|null;
|
|
|
|
templateUrl: string|null;
|
|
|
|
styles: string[];
|
|
|
|
styleUrls: string[];
|
|
|
|
interpolation: [string, string]|null;
|
|
|
|
encapsulation: ViewEncapsulation|null;
|
2017-10-04 10:51:34 -07:00
|
|
|
animations: any[];
|
2017-07-28 15:58:28 +02:00
|
|
|
preserveWhitespaces: boolean|null;
|
2016-11-10 14:07:30 -08:00
|
|
|
}
|
|
|
|
|
2016-01-06 14:13:44 -08:00
|
|
|
export class DirectiveNormalizer {
|
2017-05-17 15:39:08 -07:00
|
|
|
private _resourceLoaderCache = new Map<string, SyncAsync<string>>();
|
2016-06-24 08:46:43 -07:00
|
|
|
|
2016-06-08 16:38:52 -07:00
|
|
|
constructor(
|
2016-08-17 09:24:44 -07:00
|
|
|
private _resourceLoader: ResourceLoader, private _urlResolver: UrlResolver,
|
|
|
|
private _htmlParser: HtmlParser, private _config: CompilerConfig) {}
|
2015-08-25 15:36:02 -07:00
|
|
|
|
2016-12-11 13:43:51 +03:00
|
|
|
clearCache(): void { this._resourceLoaderCache.clear(); }
|
2016-06-24 08:46:43 -07:00
|
|
|
|
2016-12-11 13:43:51 +03:00
|
|
|
clearCacheFor(normalizedDirective: CompileDirectiveMetadata): void {
|
2016-06-24 08:46:43 -07:00
|
|
|
if (!normalizedDirective.isComponent) {
|
|
|
|
return;
|
|
|
|
}
|
2017-03-24 09:59:58 -07:00
|
|
|
const template = normalizedDirective.template !;
|
|
|
|
this._resourceLoaderCache.delete(template.templateUrl !);
|
|
|
|
template.externalStylesheets.forEach(
|
|
|
|
(stylesheet) => { this._resourceLoaderCache.delete(stylesheet.moduleUrl !); });
|
2016-06-24 08:46:43 -07:00
|
|
|
}
|
|
|
|
|
2017-05-17 15:39:08 -07:00
|
|
|
private _fetch(url: string): SyncAsync<string> {
|
2016-11-12 14:08:58 +01:00
|
|
|
let result = this._resourceLoaderCache.get(url);
|
2016-06-24 08:46:43 -07:00
|
|
|
if (!result) {
|
2017-05-17 15:39:08 -07:00
|
|
|
result = this._resourceLoader.get(url);
|
2016-08-17 09:24:44 -07:00
|
|
|
this._resourceLoaderCache.set(url, result);
|
2016-06-24 08:46:43 -07:00
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2016-11-10 14:07:30 -08:00
|
|
|
normalizeTemplate(prenormData: PrenormalizedTemplateMetadata):
|
2017-05-17 15:39:08 -07:00
|
|
|
SyncAsync<CompileTemplateMetadata> {
|
2017-03-24 09:59:58 -07:00
|
|
|
if (isDefined(prenormData.template)) {
|
|
|
|
if (isDefined(prenormData.templateUrl)) {
|
2017-03-29 20:26:48 +03:00
|
|
|
throw syntaxError(
|
|
|
|
`'${stringify(prenormData.componentType)}' component cannot define both template and templateUrl`);
|
|
|
|
}
|
2016-12-11 13:49:03 +03:00
|
|
|
if (typeof prenormData.template !== 'string') {
|
2017-01-27 13:19:00 -08:00
|
|
|
throw syntaxError(
|
2016-12-11 13:49:03 +03:00
|
|
|
`The template specified for component ${stringify(prenormData.componentType)} is not a string`);
|
|
|
|
}
|
2017-03-24 09:59:58 -07:00
|
|
|
} else if (isDefined(prenormData.templateUrl)) {
|
2016-12-11 13:49:03 +03:00
|
|
|
if (typeof prenormData.templateUrl !== 'string') {
|
2017-01-27 13:19:00 -08:00
|
|
|
throw syntaxError(
|
2016-12-11 13:49:03 +03:00
|
|
|
`The templateUrl specified for component ${stringify(prenormData.componentType)} is not a string`);
|
|
|
|
}
|
2016-06-24 08:46:43 -07:00
|
|
|
} else {
|
2017-01-27 13:19:00 -08:00
|
|
|
throw syntaxError(
|
2016-11-10 14:07:30 -08:00
|
|
|
`No template specified for component ${stringify(prenormData.componentType)}`);
|
2016-06-24 08:46:43 -07:00
|
|
|
}
|
2017-07-28 15:58:28 +02:00
|
|
|
|
|
|
|
if (isDefined(prenormData.preserveWhitespaces) &&
|
|
|
|
typeof prenormData.preserveWhitespaces !== 'boolean') {
|
|
|
|
throw syntaxError(
|
|
|
|
`The preserveWhitespaces option for component ${stringify(prenormData.componentType)} must be a boolean`);
|
|
|
|
}
|
|
|
|
|
2017-05-17 15:39:08 -07:00
|
|
|
return SyncAsync.then(
|
2017-09-12 09:40:28 -07:00
|
|
|
this._preParseTemplate(prenormData),
|
|
|
|
(preparsedTemplate) => this._normalizeTemplateMetadata(prenormData, preparsedTemplate));
|
2017-05-17 15:39:08 -07:00
|
|
|
}
|
2016-11-10 14:07:30 -08:00
|
|
|
|
2017-09-12 09:40:28 -07:00
|
|
|
private _preParseTemplate(prenomData: PrenormalizedTemplateMetadata):
|
|
|
|
SyncAsync<PreparsedTemplate> {
|
2017-05-17 15:39:08 -07:00
|
|
|
let template: SyncAsync<string>;
|
|
|
|
let templateUrl: string;
|
|
|
|
if (prenomData.template != null) {
|
|
|
|
template = prenomData.template;
|
|
|
|
templateUrl = prenomData.moduleUrl;
|
2016-06-24 08:46:43 -07:00
|
|
|
} else {
|
2017-05-17 15:39:08 -07:00
|
|
|
templateUrl = this._urlResolver.resolve(prenomData.moduleUrl, prenomData.templateUrl !);
|
|
|
|
template = this._fetch(templateUrl);
|
2016-06-24 08:46:43 -07:00
|
|
|
}
|
2017-05-17 15:39:08 -07:00
|
|
|
return SyncAsync.then(
|
2017-09-12 09:40:28 -07:00
|
|
|
template, (template) => this._preparseLoadedTemplate(prenomData, template, templateUrl));
|
2015-08-25 15:36:02 -07:00
|
|
|
}
|
|
|
|
|
2017-09-12 09:40:28 -07:00
|
|
|
private _preparseLoadedTemplate(
|
2017-03-14 09:16:15 -07:00
|
|
|
prenormData: PrenormalizedTemplateMetadata, template: string,
|
2017-09-12 09:40:28 -07:00
|
|
|
templateAbsUrl: string): PreparsedTemplate {
|
2017-03-14 09:16:15 -07:00
|
|
|
const isInline = !!prenormData.template;
|
2017-03-24 09:59:58 -07:00
|
|
|
const interpolationConfig = InterpolationConfig.fromArray(prenormData.interpolation !);
|
2016-11-10 14:07:30 -08:00
|
|
|
const rootNodesAndErrors = this._htmlParser.parse(
|
2017-03-14 09:16:15 -07:00
|
|
|
template,
|
|
|
|
templateSourceUrl(
|
|
|
|
{reference: prenormData.ngModuleType}, {type: {reference: prenormData.componentType}},
|
|
|
|
{isInline, templateUrl: templateAbsUrl}),
|
|
|
|
true, interpolationConfig);
|
2015-10-07 09:34:21 -07:00
|
|
|
if (rootNodesAndErrors.errors.length > 0) {
|
2016-07-09 10:12:39 -07:00
|
|
|
const errorString = rootNodesAndErrors.errors.join('\n');
|
2017-01-27 13:19:00 -08:00
|
|
|
throw syntaxError(`Template parse errors:\n${errorString}`);
|
2015-10-07 09:34:21 -07:00
|
|
|
}
|
2016-12-22 10:17:41 -08:00
|
|
|
|
2017-09-12 09:40:28 -07:00
|
|
|
const templateMetadataStyles = this._normalizeStylesheet(new CompileStylesheetMetadata(
|
|
|
|
{styles: prenormData.styles, moduleUrl: prenormData.moduleUrl}));
|
2015-10-07 09:34:21 -07:00
|
|
|
|
2016-07-09 10:12:39 -07:00
|
|
|
const visitor = new TemplatePreparseVisitor();
|
2016-07-21 13:56:58 -07:00
|
|
|
html.visitAll(visitor, rootNodesAndErrors.rootNodes);
|
2017-09-12 09:40:28 -07:00
|
|
|
const templateStyles = this._normalizeStylesheet(new CompileStylesheetMetadata(
|
2016-06-24 08:46:43 -07:00
|
|
|
{styles: visitor.styles, styleUrls: visitor.styleUrls, moduleUrl: templateAbsUrl}));
|
2015-09-14 15:59:09 -07:00
|
|
|
|
2017-09-12 09:40:28 -07:00
|
|
|
const styles = templateMetadataStyles.styles.concat(templateStyles.styles);
|
|
|
|
|
|
|
|
const inlineStyleUrls = templateMetadataStyles.styleUrls.concat(templateStyles.styleUrls);
|
|
|
|
const styleUrls = this
|
|
|
|
._normalizeStylesheet(new CompileStylesheetMetadata(
|
|
|
|
{styleUrls: prenormData.styleUrls, moduleUrl: prenormData.moduleUrl}))
|
|
|
|
.styleUrls;
|
|
|
|
return {
|
|
|
|
template,
|
|
|
|
templateUrl: templateAbsUrl, isInline,
|
|
|
|
htmlAst: rootNodesAndErrors, styles, inlineStyleUrls, styleUrls,
|
|
|
|
ngContentSelectors: visitor.ngContentSelectors,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
private _normalizeTemplateMetadata(
|
|
|
|
prenormData: PrenormalizedTemplateMetadata,
|
|
|
|
preparsedTemplate: PreparsedTemplate): SyncAsync<CompileTemplateMetadata> {
|
|
|
|
return SyncAsync.then(
|
|
|
|
this._loadMissingExternalStylesheets(
|
|
|
|
preparsedTemplate.styleUrls.concat(preparsedTemplate.inlineStyleUrls)),
|
|
|
|
(externalStylesheets) => this._normalizeLoadedTemplateMetadata(
|
|
|
|
prenormData, preparsedTemplate, externalStylesheets));
|
|
|
|
}
|
|
|
|
|
|
|
|
private _normalizeLoadedTemplateMetadata(
|
|
|
|
prenormData: PrenormalizedTemplateMetadata, preparsedTemplate: PreparsedTemplate,
|
|
|
|
stylesheets: Map<string, CompileStylesheetMetadata>): CompileTemplateMetadata {
|
|
|
|
// Algorithm:
|
|
|
|
// - produce exactly 1 entry per original styleUrl in
|
2018-03-10 17:14:58 +00:00
|
|
|
// CompileTemplateMetadata.externalStylesheets with all styles inlined
|
2017-09-12 09:40:28 -07:00
|
|
|
// - inline all styles that are referenced by the template into CompileTemplateMetadata.styles.
|
|
|
|
// Reason: be able to determine how many stylesheets there are even without loading
|
|
|
|
// the template nor the stylesheets, so we can create a stub for TypeScript always synchronously
|
2018-03-10 17:14:58 +00:00
|
|
|
// (as resource loading may be async)
|
2017-09-12 09:40:28 -07:00
|
|
|
|
|
|
|
const styles = [...preparsedTemplate.styles];
|
|
|
|
this._inlineStyles(preparsedTemplate.inlineStyleUrls, stylesheets, styles);
|
|
|
|
const styleUrls = preparsedTemplate.styleUrls;
|
|
|
|
|
|
|
|
const externalStylesheets = styleUrls.map(styleUrl => {
|
|
|
|
const stylesheet = stylesheets.get(styleUrl) !;
|
|
|
|
const styles = [...stylesheet.styles];
|
|
|
|
this._inlineStyles(stylesheet.styleUrls, stylesheets, styles);
|
|
|
|
return new CompileStylesheetMetadata({moduleUrl: styleUrl, styles: styles});
|
|
|
|
});
|
|
|
|
|
2017-03-14 09:16:15 -07:00
|
|
|
let encapsulation = prenormData.encapsulation;
|
2016-12-11 13:43:51 +03:00
|
|
|
if (encapsulation == null) {
|
2016-04-03 08:34:44 +09:00
|
|
|
encapsulation = this._config.defaultEncapsulation;
|
|
|
|
}
|
2016-09-20 14:14:57 -07:00
|
|
|
if (encapsulation === ViewEncapsulation.Emulated && styles.length === 0 &&
|
|
|
|
styleUrls.length === 0) {
|
2015-09-18 10:33:23 -07:00
|
|
|
encapsulation = ViewEncapsulation.None;
|
|
|
|
}
|
|
|
|
return new CompileTemplateMetadata({
|
2016-06-22 17:25:42 -07:00
|
|
|
encapsulation,
|
2017-09-12 09:40:28 -07:00
|
|
|
template: preparsedTemplate.template,
|
|
|
|
templateUrl: preparsedTemplate.templateUrl,
|
|
|
|
htmlAst: preparsedTemplate.htmlAst, styles, styleUrls,
|
|
|
|
ngContentSelectors: preparsedTemplate.ngContentSelectors,
|
2017-03-14 09:16:15 -07:00
|
|
|
animations: prenormData.animations,
|
2017-09-12 09:40:28 -07:00
|
|
|
interpolation: prenormData.interpolation,
|
|
|
|
isInline: preparsedTemplate.isInline, externalStylesheets,
|
2017-07-28 15:58:28 +02:00
|
|
|
preserveWhitespaces: preserveWhitespacesDefault(
|
|
|
|
prenormData.preserveWhitespaces, this._config.preserveWhitespaces),
|
2015-08-25 15:36:02 -07:00
|
|
|
});
|
|
|
|
}
|
2016-06-24 08:46:43 -07:00
|
|
|
|
2017-09-12 09:40:28 -07:00
|
|
|
private _inlineStyles(
|
|
|
|
styleUrls: string[], stylesheets: Map<string, CompileStylesheetMetadata>,
|
|
|
|
targetStyles: string[]) {
|
|
|
|
styleUrls.forEach(styleUrl => {
|
|
|
|
const stylesheet = stylesheets.get(styleUrl) !;
|
|
|
|
stylesheet.styles.forEach(style => targetStyles.push(style));
|
|
|
|
this._inlineStyles(stylesheet.styleUrls, stylesheets, targetStyles);
|
|
|
|
});
|
2016-06-24 08:46:43 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
private _loadMissingExternalStylesheets(
|
|
|
|
styleUrls: string[],
|
|
|
|
loadedStylesheets:
|
|
|
|
Map<string, CompileStylesheetMetadata> = new Map<string, CompileStylesheetMetadata>()):
|
2017-09-12 09:40:28 -07:00
|
|
|
SyncAsync<Map<string, CompileStylesheetMetadata>> {
|
2017-05-17 15:39:08 -07:00
|
|
|
return SyncAsync.then(
|
|
|
|
SyncAsync.all(styleUrls.filter((styleUrl) => !loadedStylesheets.has(styleUrl))
|
|
|
|
.map(
|
|
|
|
styleUrl => SyncAsync.then(
|
|
|
|
this._fetch(styleUrl),
|
|
|
|
(loadedStyle) => {
|
|
|
|
const stylesheet =
|
2017-09-12 09:40:28 -07:00
|
|
|
this._normalizeStylesheet(new CompileStylesheetMetadata(
|
2017-05-17 15:39:08 -07:00
|
|
|
{styles: [loadedStyle], moduleUrl: styleUrl}));
|
|
|
|
loadedStylesheets.set(styleUrl, stylesheet);
|
|
|
|
return this._loadMissingExternalStylesheets(
|
|
|
|
stylesheet.styleUrls, loadedStylesheets);
|
|
|
|
}))),
|
2017-09-12 09:40:28 -07:00
|
|
|
(_) => loadedStylesheets);
|
2016-06-24 08:46:43 -07:00
|
|
|
}
|
|
|
|
|
2017-09-12 09:40:28 -07:00
|
|
|
private _normalizeStylesheet(stylesheet: CompileStylesheetMetadata): CompileStylesheetMetadata {
|
2017-03-24 09:59:58 -07:00
|
|
|
const moduleUrl = stylesheet.moduleUrl !;
|
2016-11-12 14:08:58 +01:00
|
|
|
const allStyleUrls = stylesheet.styleUrls.filter(isStyleUrlResolvable)
|
2017-03-24 09:59:58 -07:00
|
|
|
.map(url => this._urlResolver.resolve(moduleUrl, url));
|
2016-06-24 08:46:43 -07:00
|
|
|
|
2016-11-12 14:08:58 +01:00
|
|
|
const allStyles = stylesheet.styles.map(style => {
|
2017-03-24 09:59:58 -07:00
|
|
|
const styleWithImports = extractStyleUrls(this._urlResolver, moduleUrl, style);
|
2016-06-24 08:46:43 -07:00
|
|
|
allStyleUrls.push(...styleWithImports.styleUrls);
|
|
|
|
return styleWithImports.style;
|
|
|
|
});
|
|
|
|
|
|
|
|
return new CompileStylesheetMetadata(
|
2017-03-24 09:59:58 -07:00
|
|
|
{styles: allStyles, styleUrls: allStyleUrls, moduleUrl: moduleUrl});
|
2016-06-24 08:46:43 -07:00
|
|
|
}
|
2015-08-25 15:36:02 -07:00
|
|
|
}
|
|
|
|
|
2017-09-12 09:40:28 -07:00
|
|
|
interface PreparsedTemplate {
|
|
|
|
template: string;
|
|
|
|
templateUrl: string;
|
|
|
|
isInline: boolean;
|
|
|
|
htmlAst: HtmlParseTreeResult;
|
|
|
|
styles: string[];
|
|
|
|
inlineStyleUrls: string[];
|
|
|
|
styleUrls: string[];
|
|
|
|
ngContentSelectors: string[];
|
|
|
|
}
|
|
|
|
|
2016-07-21 13:56:58 -07:00
|
|
|
class TemplatePreparseVisitor implements html.Visitor {
|
2015-08-25 15:36:02 -07:00
|
|
|
ngContentSelectors: string[] = [];
|
|
|
|
styles: string[] = [];
|
|
|
|
styleUrls: string[] = [];
|
2015-09-18 10:33:23 -07:00
|
|
|
ngNonBindableStackCount: number = 0;
|
2015-08-25 15:36:02 -07:00
|
|
|
|
2016-07-21 13:56:58 -07:00
|
|
|
visitElement(ast: html.Element, context: any): any {
|
2016-11-12 14:08:58 +01:00
|
|
|
const preparsedElement = preparseElement(ast);
|
2015-09-18 10:33:23 -07:00
|
|
|
switch (preparsedElement.type) {
|
|
|
|
case PreparsedElementType.NG_CONTENT:
|
2015-09-18 10:33:23 -07:00
|
|
|
if (this.ngNonBindableStackCount === 0) {
|
|
|
|
this.ngContentSelectors.push(preparsedElement.selectAttr);
|
|
|
|
}
|
2015-09-18 10:33:23 -07:00
|
|
|
break;
|
|
|
|
case PreparsedElementType.STYLE:
|
2016-11-12 14:08:58 +01:00
|
|
|
let textContent = '';
|
2015-09-18 10:33:23 -07:00
|
|
|
ast.children.forEach(child => {
|
2016-07-21 13:56:58 -07:00
|
|
|
if (child instanceof html.Text) {
|
2016-07-13 11:01:32 -07:00
|
|
|
textContent += child.value;
|
2015-09-18 10:33:23 -07:00
|
|
|
}
|
|
|
|
});
|
|
|
|
this.styles.push(textContent);
|
|
|
|
break;
|
|
|
|
case PreparsedElementType.STYLESHEET:
|
|
|
|
this.styleUrls.push(preparsedElement.hrefAttr);
|
|
|
|
break;
|
2016-01-25 17:57:29 -08:00
|
|
|
default:
|
|
|
|
break;
|
2015-08-25 15:36:02 -07:00
|
|
|
}
|
2015-09-18 10:33:23 -07:00
|
|
|
if (preparsedElement.nonBindable) {
|
|
|
|
this.ngNonBindableStackCount++;
|
|
|
|
}
|
2016-07-21 13:56:58 -07:00
|
|
|
html.visitAll(this, ast.children);
|
2015-09-18 10:33:23 -07:00
|
|
|
if (preparsedElement.nonBindable) {
|
|
|
|
this.ngNonBindableStackCount--;
|
2015-08-25 15:36:02 -07:00
|
|
|
}
|
2015-09-18 10:33:23 -07:00
|
|
|
return null;
|
2015-08-25 15:36:02 -07:00
|
|
|
}
|
2016-08-11 21:00:35 -07:00
|
|
|
|
2016-12-22 10:17:41 -08:00
|
|
|
visitExpansion(ast: html.Expansion, context: any): any { html.visitAll(this, ast.cases); }
|
|
|
|
|
|
|
|
visitExpansionCase(ast: html.ExpansionCase, context: any): any {
|
|
|
|
html.visitAll(this, ast.expression);
|
|
|
|
}
|
|
|
|
|
2016-07-21 13:56:58 -07:00
|
|
|
visitComment(ast: html.Comment, context: any): any { return null; }
|
|
|
|
visitAttribute(ast: html.Attribute, context: any): any { return null; }
|
|
|
|
visitText(ast: html.Text, context: any): any { return null; }
|
2015-08-25 15:36:02 -07:00
|
|
|
}
|