refactor(template): remove supporter deprecated `var` / `#` (#11084)

BREAKING CHANGES:

- `#` and `var` are not supported any more in expressions, use `let`,
- `var-<name>` could not be used any more on templates, use `let-<name>`,
- `var-<name>` could not be used any more to create a reference, use `ref-<name>`.
This commit is contained in:
Victor Berchet 2016-08-25 15:21:33 -07:00 committed by GitHub
parent ce08982f78
commit b867764b0d
4 changed files with 68 additions and 128 deletions

View File

@ -237,16 +237,11 @@ export class _ParseAST {
peekKeywordLet(): boolean { return this.next.isKeywordLet(); } peekKeywordLet(): boolean { return this.next.isKeywordLet(); }
peekDeprecatedKeywordVar(): boolean { return this.next.isKeywordDeprecatedVar(); }
peekDeprecatedOperatorHash(): boolean { return this.next.isOperator('#'); }
expectCharacter(code: number) { expectCharacter(code: number) {
if (this.optionalCharacter(code)) return; if (this.optionalCharacter(code)) return;
this.error(`Missing expected ${StringWrapper.fromCharCode(code)}`); this.error(`Missing expected ${StringWrapper.fromCharCode(code)}`);
} }
optionalOperator(op: string): boolean { optionalOperator(op: string): boolean {
if (this.next.isOperator(op)) { if (this.next.isOperator(op)) {
this.advance(); this.advance();
@ -659,15 +654,7 @@ export class _ParseAST {
let prefix: string = null; let prefix: string = null;
let warnings: string[] = []; let warnings: string[] = [];
while (this.index < this.tokens.length) { while (this.index < this.tokens.length) {
var keyIsVar: boolean = this.peekKeywordLet(); const keyIsVar: boolean = this.peekKeywordLet();
if (!keyIsVar && this.peekDeprecatedKeywordVar()) {
keyIsVar = true;
warnings.push(`"var" inside of expressions is deprecated. Use "let" instead!`);
}
if (!keyIsVar && this.peekDeprecatedOperatorHash()) {
keyIsVar = true;
warnings.push(`"#" inside of expressions is deprecated. Use "let" instead!`);
}
if (keyIsVar) { if (keyIsVar) {
this.advance(); this.advance();
} }
@ -688,12 +675,10 @@ export class _ParseAST {
} else { } else {
name = '\$implicit'; name = '\$implicit';
} }
} else if ( } else if (this.next !== EOF && !this.peekKeywordLet()) {
this.next !== EOF && !this.peekKeywordLet() && !this.peekDeprecatedKeywordVar() &&
!this.peekDeprecatedOperatorHash()) {
const start = this.inputIndex; const start = this.inputIndex;
var ast = this.parsePipe(); const ast = this.parsePipe();
var source = this.input.substring(start, this.inputIndex); const source = this.input.substring(start, this.inputIndex);
expression = new ASTWithSource(ast, source, this.location, this.errors); expression = new ASTWithSource(ast, source, this.location, this.errors);
} }
bindings.push(new TemplateBinding(key, keyIsVar, name, expression)); bindings.push(new TemplateBinding(key, keyIsVar, name, expression));
@ -723,7 +708,7 @@ export class _ParseAST {
// of the '(' begins an '(' <expr> ')' production). The recovery points of grouping symbols // of the '(' begins an '(' <expr> ')' production). The recovery points of grouping symbols
// must be conditional as they must be skipped if none of the calling productions are not // must be conditional as they must be skipped if none of the calling productions are not
// expecting the closing token else we will never make progress in the case of an // expecting the closing token else we will never make progress in the case of an
// extrainious group closing symbol (such as a stray ')'). This is not the case for ';' because // extraneous group closing symbol (such as a stray ')'). This is not the case for ';' because
// parseChain() is always the root production and it expects a ';'. // parseChain() is always the root production and it expects a ';'.
// If a production expects one of these token it increments the corresponding nesting count, // If a production expects one of these token it increments the corresponding nesting count,

View File

@ -34,18 +34,28 @@ import {PreparsedElementType, preparseElement} from './template_preparser';
// Group 1 = "bind-" // Group 1 = "bind-"
// Group 2 = "var-" // Group 2 = "let-"
// Group 3 = "let-" // Group 3 = "ref-/#"
// Group 4 = "ref-/#" // Group 4 = "on-"
// Group 5 = "on-" // Group 5 = "bindon-"
// Group 6 = "bindon-" // Group 6 = "@"
// Group 7 = "@" // Group 7 = the identifier after "bind-", "let-", "ref-/#", "on-", "bindon-" or "@"
// Group 8 = the identifier after "bind-", "var-/#", or "on-" // Group 8 = identifier inside [()]
// Group 9 = identifier inside [()] // Group 9 = identifier inside []
// Group 10 = identifier inside [] // Group 10 = identifier inside ()
// Group 11 = identifier inside ()
const BIND_NAME_REGEXP = const BIND_NAME_REGEXP =
/^(?:(?:(?:(bind-)|(var-)|(let-)|(ref-|#)|(on-)|(bindon-)|(@))(.+))|\[\(([^\)]+)\)\]|\[([^\]]+)\]|\(([^\)]+)\))$/; /^(?:(?:(?:(bind-)|(let-)|(ref-|#)|(on-)|(bindon-)|(@))(.+))|\[\(([^\)]+)\)\]|\[([^\]]+)\]|\(([^\)]+)\))$/;
const KW_BIND_IDX = 1;
const KW_LET_IDX = 2;
const KW_REF_IDX = 3;
const KW_ON_IDX = 4;
const KW_BINDON_IDX = 5;
const KW_AT_IDX = 6;
const IDENT_KW_IDX = 7;
const IDENT_BANANA_BOX_IDX = 8;
const IDENT_PROPERTY_IDX = 9;
const IDENT_EVENT_IDX = 10;
const ANIMATE_PROP_PREFIX = 'animate-'; const ANIMATE_PROP_PREFIX = 'animate-';
const TEMPLATE_ELEMENT = 'template'; const TEMPLATE_ELEMENT = 'template';
@ -490,90 +500,82 @@ class TemplateParseVisitor implements html.Visitor {
targetProps: BoundElementOrDirectiveProperty[], targetProps: BoundElementOrDirectiveProperty[],
targetAnimationProps: BoundElementPropertyAst[], targetEvents: BoundEventAst[], targetAnimationProps: BoundElementPropertyAst[], targetEvents: BoundEventAst[],
targetRefs: ElementOrDirectiveRef[], targetVars: VariableAst[]): boolean { targetRefs: ElementOrDirectiveRef[], targetVars: VariableAst[]): boolean {
const attrName = this._normalizeAttributeName(attr.name); const name = this._normalizeAttributeName(attr.name);
const attrValue = attr.value; const value = attr.value;
const bindParts = attrName.match(BIND_NAME_REGEXP); const srcSpan = attr.sourceSpan;
const bindParts = name.match(BIND_NAME_REGEXP);
let hasBinding = false; let hasBinding = false;
if (bindParts !== null) { if (bindParts !== null) {
hasBinding = true; hasBinding = true;
if (isPresent(bindParts[1])) { // match: bind-prop if (isPresent(bindParts[KW_BIND_IDX])) {
this._parsePropertyOrAnimation( this._parsePropertyOrAnimation(
bindParts[8], attrValue, attr.sourceSpan, targetMatchableAttrs, targetProps, bindParts[IDENT_KW_IDX], value, srcSpan, targetMatchableAttrs, targetProps,
targetAnimationProps); targetAnimationProps);
} else if (isPresent(bindParts[2])) { // match: var-name / var-name="iden" } else if (bindParts[KW_LET_IDX]) {
const identifier = bindParts[8];
if (isTemplateElement) { if (isTemplateElement) {
this._reportError( const identifier = bindParts[IDENT_KW_IDX];
`"var-" on <template> elements is deprecated. Use "let-" instead!`, attr.sourceSpan, this._parseVariable(identifier, value, srcSpan, targetVars);
ParseErrorLevel.WARNING);
this._parseVariable(identifier, attrValue, attr.sourceSpan, targetVars);
} else { } else {
this._reportError( this._reportError(`"let-" is only supported on template elements.`, srcSpan);
`"var-" on non <template> elements is deprecated. Use "ref-" instead!`,
attr.sourceSpan, ParseErrorLevel.WARNING);
this._parseReference(identifier, attrValue, attr.sourceSpan, targetRefs);
} }
} else if (isPresent(bindParts[3])) { // match: let-name } else if (bindParts[KW_REF_IDX]) {
if (isTemplateElement) { const identifier = bindParts[IDENT_KW_IDX];
const identifier = bindParts[8]; this._parseReference(identifier, value, srcSpan, targetRefs);
this._parseVariable(identifier, attrValue, attr.sourceSpan, targetVars);
} else {
this._reportError(`"let-" is only supported on template elements.`, attr.sourceSpan);
}
} else if (isPresent(bindParts[4])) { // match: ref- / #iden } else if (bindParts[KW_ON_IDX]) {
const identifier = bindParts[8];
this._parseReference(identifier, attrValue, attr.sourceSpan, targetRefs);
} else if (isPresent(bindParts[5])) { // match: on-event
this._parseEvent( this._parseEvent(
bindParts[8], attrValue, attr.sourceSpan, targetMatchableAttrs, targetEvents); bindParts[IDENT_KW_IDX], value, srcSpan, targetMatchableAttrs, targetEvents);
} else if (isPresent(bindParts[6])) { // match: bindon-prop } else if (bindParts[KW_BINDON_IDX]) {
this._parsePropertyOrAnimation( this._parsePropertyOrAnimation(
bindParts[8], attrValue, attr.sourceSpan, targetMatchableAttrs, targetProps, bindParts[IDENT_KW_IDX], value, srcSpan, targetMatchableAttrs, targetProps,
targetAnimationProps); targetAnimationProps);
this._parseAssignmentEvent( this._parseAssignmentEvent(
bindParts[8], attrValue, attr.sourceSpan, targetMatchableAttrs, targetEvents); bindParts[IDENT_KW_IDX], value, srcSpan, targetMatchableAttrs, targetEvents);
} else if (isPresent(bindParts[7])) { // match: animate-name } else if (bindParts[KW_AT_IDX]) {
if (attrName[0] == '@' && isPresent(attrValue) && attrValue.length > 0) { if (name[0] == '@' && isPresent(value) && value.length > 0) {
this._reportError( this._reportError(
`Assigning animation triggers via @prop="exp" attributes with an expression is invalid. Use property bindings (e.g. [@prop]="exp") or use an attribute without a value \(e.g. @prop\) instead.`, `Assigning animation triggers via @prop="exp" attributes with an expression is invalid.` +
attr.sourceSpan, ParseErrorLevel.FATAL); ` Use property bindings (e.g. [@prop]="exp") or use an attribute without a value (e.g. @prop) instead.`,
srcSpan, ParseErrorLevel.FATAL);
} }
this._parseAnimation( this._parseAnimation(
bindParts[8], attrValue, attr.sourceSpan, targetMatchableAttrs, targetAnimationProps); bindParts[IDENT_KW_IDX], value, srcSpan, targetMatchableAttrs, targetAnimationProps);
} else if (isPresent(bindParts[9])) { // match: [(expr)] } else if (bindParts[IDENT_BANANA_BOX_IDX]) {
this._parsePropertyOrAnimation( this._parsePropertyOrAnimation(
bindParts[9], attrValue, attr.sourceSpan, targetMatchableAttrs, targetProps, bindParts[IDENT_BANANA_BOX_IDX], value, srcSpan, targetMatchableAttrs, targetProps,
targetAnimationProps); targetAnimationProps);
this._parseAssignmentEvent( this._parseAssignmentEvent(
bindParts[9], attrValue, attr.sourceSpan, targetMatchableAttrs, targetEvents); bindParts[IDENT_BANANA_BOX_IDX], value, srcSpan, targetMatchableAttrs, targetEvents);
} else if (isPresent(bindParts[10])) { // match: [expr] } else if (bindParts[IDENT_PROPERTY_IDX]) {
this._parsePropertyOrAnimation( this._parsePropertyOrAnimation(
bindParts[10], attrValue, attr.sourceSpan, targetMatchableAttrs, targetProps, bindParts[IDENT_PROPERTY_IDX], value, srcSpan, targetMatchableAttrs, targetProps,
targetAnimationProps); targetAnimationProps);
} else if (isPresent(bindParts[11])) { // match: (event) } else if (bindParts[IDENT_EVENT_IDX]) {
this._parseEvent( this._parseEvent(
bindParts[11], attrValue, attr.sourceSpan, targetMatchableAttrs, targetEvents); bindParts[IDENT_EVENT_IDX], value, srcSpan, targetMatchableAttrs, targetEvents);
} }
} else { } else {
hasBinding = this._parsePropertyInterpolation( hasBinding =
attrName, attrValue, attr.sourceSpan, targetMatchableAttrs, targetProps); this._parsePropertyInterpolation(name, value, srcSpan, targetMatchableAttrs, targetProps);
} }
if (!hasBinding) { if (!hasBinding) {
this._parseLiteralAttr(attrName, attrValue, attr.sourceSpan, targetProps); this._parseLiteralAttr(name, value, srcSpan, targetProps);
} }
return hasBinding; return hasBinding;
} }
private _normalizeAttributeName(attrName: string): string { private _normalizeAttributeName(attrName: string): string {
return attrName.toLowerCase().startsWith('data-') ? attrName.substring(5) : attrName; return /^data-/i.test(attrName) ? attrName.substring(5) : attrName;
} }
private _parseVariable( private _parseVariable(

View File

@ -389,22 +389,6 @@ export function main() {
expect(bindings[0].expression.location).toEqual('location'); expect(bindings[0].expression.location).toEqual('location');
}); });
it('should support var notation with a deprecation warning', () => {
var bindings = createParser().parseTemplateBindings('var i', null);
expect(keyValues(bindings.templateBindings)).toEqual(['let i=\$implicit']);
expect(bindings.warnings).toEqual([
'"var" inside of expressions is deprecated. Use "let" instead!'
]);
});
it('should support # notation with a deprecation warning', () => {
var bindings = createParser().parseTemplateBindings('#i', null);
expect(keyValues(bindings.templateBindings)).toEqual(['let i=\$implicit']);
expect(bindings.warnings).toEqual([
'"#" inside of expressions is deprecated. Use "let" instead!'
]);
});
it('should support let notation', () => { it('should support let notation', () => {
var bindings = parseTemplateBindings('let i'); var bindings = parseTemplateBindings('let i');
expect(keyValues(bindings)).toEqual(['let i=\$implicit']); expect(keyValues(bindings)).toEqual(['let i=\$implicit']);

View File

@ -838,15 +838,6 @@ Can't bind to 'invalidProp' since it isn't a known property of 'my-component'.
]))).toEqual([[ElementAst, 'div'], [ReferenceAst, 'a', null]]); ]))).toEqual([[ElementAst, 'div'], [ReferenceAst, 'a', null]]);
}); });
it('should parse references via var-... and report them as deprecated', () => {
expect(humanizeTplAst(parse('<div var-a>', [
]))).toEqual([[ElementAst, 'div'], [ReferenceAst, 'a', null]]);
expect(console.warnings).toEqual([[
'Template parse warnings:',
'"var-" on non <template> elements is deprecated. Use "ref-" instead! ("<div [ERROR ->]var-a>"): TestComp@0:5'
].join('\n')]);
});
it('should parse camel case references', () => { it('should parse camel case references', () => {
expect(humanizeTplAst(parse('<div ref-someA>', [ expect(humanizeTplAst(parse('<div ref-someA>', [
]))).toEqual([[ElementAst, 'div'], [ReferenceAst, 'someA', null]]); ]))).toEqual([[ElementAst, 'div'], [ReferenceAst, 'someA', null]]);
@ -960,15 +951,6 @@ Reference "#a" is defined several times ("<div #a></div><div [ERROR ->]#a></div>
]))).toEqual([[EmbeddedTemplateAst], [VariableAst, 'a', 'b']]); ]))).toEqual([[EmbeddedTemplateAst], [VariableAst, 'a', 'b']]);
}); });
it('should parse variables via var-... and report them as deprecated', () => {
expect(humanizeTplAst(parse('<template var-a="b">', [
]))).toEqual([[EmbeddedTemplateAst], [VariableAst, 'a', 'b']]);
expect(console.warnings).toEqual([[
'Template parse warnings:',
'"var-" on <template> elements is deprecated. Use "let-" instead! ("<template [ERROR ->]var-a="b">"): TestComp@0:10'
].join('\n')]);
});
it('should not locate directives in variables', () => { it('should not locate directives in variables', () => {
var dirA = CompileDirectiveMetadata.create({ var dirA = CompileDirectiveMetadata.create({
selector: '[a]', selector: '[a]',
@ -1000,22 +982,9 @@ Reference "#a" is defined several times ("<div #a></div><div [ERROR ->]#a></div>
]); ]);
}); });
it('should parse variables via #... and report them as deprecated', () => { it('should report an error on variables declared with #', () => {
expect(humanizeTplAst(parse('<div *ngIf="#a=b">', [ expect(() => humanizeTplAst(parse('<div *ngIf="#a=b">', [])))
]))).toEqual([[EmbeddedTemplateAst], [VariableAst, 'a', 'b'], [ElementAst, 'div']]); .toThrowError(/Parser Error: Unexpected token # at column 6/);
expect(console.warnings).toEqual([[
'Template parse warnings:',
'"#" inside of expressions is deprecated. Use "let" instead! ("<div [ERROR ->]*ngIf="#a=b">"): TestComp@0:5'
].join('\n')]);
});
it('should parse variables via var ... and report them as deprecated', () => {
expect(humanizeTplAst(parse('<div *ngIf="var a=b">', [
]))).toEqual([[EmbeddedTemplateAst], [VariableAst, 'a', 'b'], [ElementAst, 'div']]);
expect(console.warnings).toEqual([[
'Template parse warnings:',
'"var" inside of expressions is deprecated. Use "let" instead! ("<div [ERROR ->]*ngIf="var a=b">"): TestComp@0:5'
].join('\n')]);
}); });
it('should parse variables via let ...', () => { it('should parse variables via let ...', () => {