feat(Compiler): Allow overriding the projection selector

fixes #6303

BREAKING CHANGE:

For static content projection, elements with *-directives are now matched against the element itself vs the template before.

    <p *ngIf="condition" foo></p>

Before:

    // Use the implicit template for projection
    <ng-content select="template"></ng-content>

After:

    // Use the actual element for projection
    <ng-content select="p[foo]"></ng-content>
Closes #7742
This commit is contained in:
Victor Berchet 2016-03-23 14:15:05 -07:00
parent 3e593b8221
commit aa966f5de2
4 changed files with 66 additions and 13 deletions

View File

@ -264,29 +264,40 @@ class TemplateParseVisitor implements HtmlAstVisitor {
this._createElementPropertyAsts(element.name, elementOrDirectiveProps, directives); this._createElementPropertyAsts(element.name, elementOrDirectiveProps, directives);
var children = htmlVisitAll(preparsedElement.nonBindable ? NON_BINDABLE_VISITOR : this, var children = htmlVisitAll(preparsedElement.nonBindable ? NON_BINDABLE_VISITOR : this,
element.children, Component.create(directives)); element.children, Component.create(directives));
var elementNgContentIndex =
hasInlineTemplates ? null : component.findNgContentIndex(elementCssSelector); // Override the actual selector when the `ngProjectAs` attribute is provided
var projectionSelector = isPresent(preparsedElement.projectAs) ?
CssSelector.parse(preparsedElement.projectAs)[0] :
elementCssSelector;
var ngContentIndex = component.findNgContentIndex(projectionSelector);
var parsedElement; var parsedElement;
if (preparsedElement.type === PreparsedElementType.NG_CONTENT) { if (preparsedElement.type === PreparsedElementType.NG_CONTENT) {
if (isPresent(element.children) && element.children.length > 0) { if (isPresent(element.children) && element.children.length > 0) {
this._reportError( this._reportError(
`<ng-content> element cannot have content. <ng-content> must be immediately followed by </ng-content>`, `<ng-content> element cannot have content. <ng-content> must be immediately followed by </ng-content>`,
element.sourceSpan); element.sourceSpan);
} }
parsedElement =
new NgContentAst(this.ngContentCount++, elementNgContentIndex, element.sourceSpan); parsedElement = new NgContentAst(
this.ngContentCount++, hasInlineTemplates ? null : ngContentIndex, element.sourceSpan);
} else if (isTemplateElement) { } else if (isTemplateElement) {
this._assertAllEventsPublishedByDirectives(directives, events); this._assertAllEventsPublishedByDirectives(directives, events);
this._assertNoComponentsNorElementBindingsOnTemplate(directives, elementProps, this._assertNoComponentsNorElementBindingsOnTemplate(directives, elementProps,
element.sourceSpan); element.sourceSpan);
parsedElement = new EmbeddedTemplateAst(attrs, events, vars, directives, children,
elementNgContentIndex, element.sourceSpan); parsedElement =
new EmbeddedTemplateAst(attrs, events, vars, directives, children,
hasInlineTemplates ? null : ngContentIndex, element.sourceSpan);
} else { } else {
this._assertOnlyOneComponent(directives, element.sourceSpan); this._assertOnlyOneComponent(directives, element.sourceSpan);
var elementExportAsVars = vars.filter(varAst => varAst.value.length === 0); var elementExportAsVars = vars.filter(varAst => varAst.value.length === 0);
let ngContentIndex =
hasInlineTemplates ? null : component.findNgContentIndex(projectionSelector);
parsedElement = parsedElement =
new ElementAst(nodeName, attrs, elementProps, events, elementExportAsVars, directives, new ElementAst(nodeName, attrs, elementProps, events, elementExportAsVars, directives,
children, elementNgContentIndex, element.sourceSpan); children, hasInlineTemplates ? null : ngContentIndex, element.sourceSpan);
} }
if (hasInlineTemplates) { if (hasInlineTemplates) {
var templateCssSelector = createElementCssSelector(TEMPLATE_ELEMENT, templateMatchableAttrs); var templateCssSelector = createElementCssSelector(TEMPLATE_ELEMENT, templateMatchableAttrs);
@ -297,9 +308,9 @@ class TemplateParseVisitor implements HtmlAstVisitor {
element.name, templateElementOrDirectiveProps, templateDirectives); element.name, templateElementOrDirectiveProps, templateDirectives);
this._assertNoComponentsNorElementBindingsOnTemplate(templateDirectives, templateElementProps, this._assertNoComponentsNorElementBindingsOnTemplate(templateDirectives, templateElementProps,
element.sourceSpan); element.sourceSpan);
parsedElement = new EmbeddedTemplateAst(
[], [], templateVars, templateDirectives, [parsedElement], parsedElement = new EmbeddedTemplateAst([], [], templateVars, templateDirectives,
component.findNgContentIndex(templateCssSelector), element.sourceSpan); [parsedElement], ngContentIndex, element.sourceSpan);
} }
return parsedElement; return parsedElement;
} }

View File

@ -11,12 +11,14 @@ const LINK_STYLE_REL_VALUE = 'stylesheet';
const STYLE_ELEMENT = 'style'; const STYLE_ELEMENT = 'style';
const SCRIPT_ELEMENT = 'script'; const SCRIPT_ELEMENT = 'script';
const NG_NON_BINDABLE_ATTR = 'ngNonBindable'; const NG_NON_BINDABLE_ATTR = 'ngNonBindable';
const NG_PROJECT_AS = 'ngProjectAs';
export function preparseElement(ast: HtmlElementAst): PreparsedElement { export function preparseElement(ast: HtmlElementAst): PreparsedElement {
var selectAttr = null; var selectAttr = null;
var hrefAttr = null; var hrefAttr = null;
var relAttr = null; var relAttr = null;
var nonBindable = false; var nonBindable = false;
var projectAs: string = null;
ast.attrs.forEach(attr => { ast.attrs.forEach(attr => {
let lcAttrName = attr.name.toLowerCase(); let lcAttrName = attr.name.toLowerCase();
if (lcAttrName == NG_CONTENT_SELECT_ATTR) { if (lcAttrName == NG_CONTENT_SELECT_ATTR) {
@ -27,6 +29,10 @@ export function preparseElement(ast: HtmlElementAst): PreparsedElement {
relAttr = attr.value; relAttr = attr.value;
} else if (attr.name == NG_NON_BINDABLE_ATTR) { } else if (attr.name == NG_NON_BINDABLE_ATTR) {
nonBindable = true; nonBindable = true;
} else if (attr.name == NG_PROJECT_AS) {
if (attr.value.length > 0) {
projectAs = attr.value;
}
} }
}); });
selectAttr = normalizeNgContentSelect(selectAttr); selectAttr = normalizeNgContentSelect(selectAttr);
@ -41,7 +47,7 @@ export function preparseElement(ast: HtmlElementAst): PreparsedElement {
} else if (nodeName == LINK_ELEMENT && relAttr == LINK_STYLE_REL_VALUE) { } else if (nodeName == LINK_ELEMENT && relAttr == LINK_STYLE_REL_VALUE) {
type = PreparsedElementType.STYLESHEET; type = PreparsedElementType.STYLESHEET;
} }
return new PreparsedElement(type, selectAttr, hrefAttr, nonBindable); return new PreparsedElement(type, selectAttr, hrefAttr, nonBindable, projectAs);
} }
export enum PreparsedElementType { export enum PreparsedElementType {
@ -54,7 +60,7 @@ export enum PreparsedElementType {
export class PreparsedElement { export class PreparsedElement {
constructor(public type: PreparsedElementType, public selectAttr: string, public hrefAttr: string, constructor(public type: PreparsedElementType, public selectAttr: string, public hrefAttr: string,
public nonBindable: boolean) {} public nonBindable: boolean, public projectAs: string) {}
} }

View File

@ -686,6 +686,39 @@ There is no directive with "exportAs" set to "dirA" ("<div [ERROR ->]#a="dirA"><
[createComp('div', ['*'])]))) [createComp('div', ['*'])])))
.toEqual([['div', null], ['#text({{hello}})', 0], ['span', 0]]); .toEqual([['div', null], ['#text({{hello}})', 0], ['span', 0]]);
}); });
it('should match the element when there is an inline template', () => {
expect(humanizeContentProjection(
parse('<div><b *ngIf="cond"></b></div>', [createComp('div', ['a', 'b']), ngIf])))
.toEqual([['div', null], ['template', 1], ['b', null]]);
});
describe('ngProjectAs', () => {
it('should override elements', () => {
expect(humanizeContentProjection(
parse('<div><a ngProjectAs="b"></a></div>', [createComp('div', ['a', 'b'])])))
.toEqual([['div', null], ['a', 1]]);
});
it('should override <ng-content>', () => {
expect(humanizeContentProjection(
parse('<div><ng-content ngProjectAs="b"></ng-content></div>',
[createComp('div', ['ng-content', 'b'])])))
.toEqual([['div', null], ['ng-content', 1]]);
});
it('should override <template>', () => {
expect(humanizeContentProjection(parse('<div><template ngProjectAs="b"></template></div>',
[createComp('div', ['template', 'b'])])))
.toEqual([['div', null], ['template', 1]]);
});
it('should override inline templates', () => {
expect(humanizeContentProjection(parse('<div><a *ngIf="cond" ngProjectAs="b"></a></div>',
[createComp('div', ['a', 'b']), ngIf])))
.toEqual([['div', null], ['template', 1], ['a', null]]);
});
});
}); });
describe('splitClasses', () => { describe('splitClasses', () => {

View File

@ -26,7 +26,7 @@ export function main() {
beforeEach(inject([HtmlParser], (_htmlParser: HtmlParser) => { htmlParser = _htmlParser; })); beforeEach(inject([HtmlParser], (_htmlParser: HtmlParser) => { htmlParser = _htmlParser; }));
function preparse(html: string): PreparsedElement { function preparse(html: string): PreparsedElement {
return preparseElement(htmlParser.parse(html, '').rootNodes[0]); return preparseElement(htmlParser.parse(html, 'TestComp').rootNodes[0]);
} }
it('should detect script elements', inject([HtmlParser], (htmlParser: HtmlParser) => { it('should detect script elements', inject([HtmlParser], (htmlParser: HtmlParser) => {
@ -54,5 +54,8 @@ export function main() {
expect(preparse('<ng-content select="*">').selectAttr).toEqual('*'); expect(preparse('<ng-content select="*">').selectAttr).toEqual('*');
})); }));
it('should extract ngProjectAs value', () => {
expect(preparse('<p ngProjectAs="el[attr].class"></p>').projectAs).toEqual('el[attr].class');
});
}); });
} }