1 | /**
|
---|
2 | * @license
|
---|
3 | * Copyright Google LLC All Rights Reserved.
|
---|
4 | *
|
---|
5 | * Use of this source code is governed by an MIT-style license that can be
|
---|
6 | * found in the LICENSE file at https://angular.io/license
|
---|
7 | */
|
---|
8 | (function (factory) {
|
---|
9 | if (typeof module === "object" && typeof module.exports === "object") {
|
---|
10 | var v = factory(require, exports);
|
---|
11 | if (v !== undefined) module.exports = v;
|
---|
12 | }
|
---|
13 | else if (typeof define === "function" && define.amd) {
|
---|
14 | define("@angular/compiler/src/template_parser/binding_parser", ["require", "exports", "tslib", "@angular/compiler/src/core", "@angular/compiler/src/expression_parser/ast", "@angular/compiler/src/ml_parser/tags", "@angular/compiler/src/parse_util", "@angular/compiler/src/selector", "@angular/compiler/src/util"], factory);
|
---|
15 | }
|
---|
16 | })(function (require, exports) {
|
---|
17 | "use strict";
|
---|
18 | Object.defineProperty(exports, "__esModule", { value: true });
|
---|
19 | exports.calcPossibleSecurityContexts = exports.PipeCollector = exports.BindingParser = void 0;
|
---|
20 | var tslib_1 = require("tslib");
|
---|
21 | var core_1 = require("@angular/compiler/src/core");
|
---|
22 | var ast_1 = require("@angular/compiler/src/expression_parser/ast");
|
---|
23 | var tags_1 = require("@angular/compiler/src/ml_parser/tags");
|
---|
24 | var parse_util_1 = require("@angular/compiler/src/parse_util");
|
---|
25 | var selector_1 = require("@angular/compiler/src/selector");
|
---|
26 | var util_1 = require("@angular/compiler/src/util");
|
---|
27 | var PROPERTY_PARTS_SEPARATOR = '.';
|
---|
28 | var ATTRIBUTE_PREFIX = 'attr';
|
---|
29 | var CLASS_PREFIX = 'class';
|
---|
30 | var STYLE_PREFIX = 'style';
|
---|
31 | var TEMPLATE_ATTR_PREFIX = '*';
|
---|
32 | var ANIMATE_PROP_PREFIX = 'animate-';
|
---|
33 | /**
|
---|
34 | * Parses bindings in templates and in the directive host area.
|
---|
35 | */
|
---|
36 | var BindingParser = /** @class */ (function () {
|
---|
37 | function BindingParser(_exprParser, _interpolationConfig, _schemaRegistry, pipes, errors) {
|
---|
38 | this._exprParser = _exprParser;
|
---|
39 | this._interpolationConfig = _interpolationConfig;
|
---|
40 | this._schemaRegistry = _schemaRegistry;
|
---|
41 | this.errors = errors;
|
---|
42 | this.pipesByName = null;
|
---|
43 | this._usedPipes = new Map();
|
---|
44 | // When the `pipes` parameter is `null`, do not check for used pipes
|
---|
45 | // This is used in IVY when we might not know the available pipes at compile time
|
---|
46 | if (pipes) {
|
---|
47 | var pipesByName_1 = new Map();
|
---|
48 | pipes.forEach(function (pipe) { return pipesByName_1.set(pipe.name, pipe); });
|
---|
49 | this.pipesByName = pipesByName_1;
|
---|
50 | }
|
---|
51 | }
|
---|
52 | Object.defineProperty(BindingParser.prototype, "interpolationConfig", {
|
---|
53 | get: function () {
|
---|
54 | return this._interpolationConfig;
|
---|
55 | },
|
---|
56 | enumerable: false,
|
---|
57 | configurable: true
|
---|
58 | });
|
---|
59 | BindingParser.prototype.getUsedPipes = function () {
|
---|
60 | return Array.from(this._usedPipes.values());
|
---|
61 | };
|
---|
62 | BindingParser.prototype.createBoundHostProperties = function (dirMeta, sourceSpan) {
|
---|
63 | var _this = this;
|
---|
64 | if (dirMeta.hostProperties) {
|
---|
65 | var boundProps_1 = [];
|
---|
66 | Object.keys(dirMeta.hostProperties).forEach(function (propName) {
|
---|
67 | var expression = dirMeta.hostProperties[propName];
|
---|
68 | if (typeof expression === 'string') {
|
---|
69 | _this.parsePropertyBinding(propName, expression, true, sourceSpan, sourceSpan.start.offset, undefined, [],
|
---|
70 | // Use the `sourceSpan` for `keySpan`. This isn't really accurate, but neither is the
|
---|
71 | // sourceSpan, as it represents the sourceSpan of the host itself rather than the
|
---|
72 | // source of the host binding (which doesn't exist in the template). Regardless,
|
---|
73 | // neither of these values are used in Ivy but are only here to satisfy the function
|
---|
74 | // signature. This should likely be refactored in the future so that `sourceSpan`
|
---|
75 | // isn't being used inaccurately.
|
---|
76 | boundProps_1, sourceSpan);
|
---|
77 | }
|
---|
78 | else {
|
---|
79 | _this._reportError("Value of the host property binding \"" + propName + "\" needs to be a string representing an expression but got \"" + expression + "\" (" + typeof expression + ")", sourceSpan);
|
---|
80 | }
|
---|
81 | });
|
---|
82 | return boundProps_1;
|
---|
83 | }
|
---|
84 | return null;
|
---|
85 | };
|
---|
86 | BindingParser.prototype.createDirectiveHostPropertyAsts = function (dirMeta, elementSelector, sourceSpan) {
|
---|
87 | var _this = this;
|
---|
88 | var boundProps = this.createBoundHostProperties(dirMeta, sourceSpan);
|
---|
89 | return boundProps &&
|
---|
90 | boundProps.map(function (prop) { return _this.createBoundElementProperty(elementSelector, prop); });
|
---|
91 | };
|
---|
92 | BindingParser.prototype.createDirectiveHostEventAsts = function (dirMeta, sourceSpan) {
|
---|
93 | var _this = this;
|
---|
94 | if (dirMeta.hostListeners) {
|
---|
95 | var targetEvents_1 = [];
|
---|
96 | Object.keys(dirMeta.hostListeners).forEach(function (propName) {
|
---|
97 | var expression = dirMeta.hostListeners[propName];
|
---|
98 | if (typeof expression === 'string') {
|
---|
99 | // Use the `sourceSpan` for `keySpan` and `handlerSpan`. This isn't really accurate, but
|
---|
100 | // neither is the `sourceSpan`, as it represents the `sourceSpan` of the host itself
|
---|
101 | // rather than the source of the host binding (which doesn't exist in the template).
|
---|
102 | // Regardless, neither of these values are used in Ivy but are only here to satisfy the
|
---|
103 | // function signature. This should likely be refactored in the future so that `sourceSpan`
|
---|
104 | // isn't being used inaccurately.
|
---|
105 | _this.parseEvent(propName, expression, sourceSpan, sourceSpan, [], targetEvents_1, sourceSpan);
|
---|
106 | }
|
---|
107 | else {
|
---|
108 | _this._reportError("Value of the host listener \"" + propName + "\" needs to be a string representing an expression but got \"" + expression + "\" (" + typeof expression + ")", sourceSpan);
|
---|
109 | }
|
---|
110 | });
|
---|
111 | return targetEvents_1;
|
---|
112 | }
|
---|
113 | return null;
|
---|
114 | };
|
---|
115 | BindingParser.prototype.parseInterpolation = function (value, sourceSpan) {
|
---|
116 | var sourceInfo = sourceSpan.start.toString();
|
---|
117 | var absoluteOffset = sourceSpan.fullStart.offset;
|
---|
118 | try {
|
---|
119 | var ast = this._exprParser.parseInterpolation(value, sourceInfo, absoluteOffset, this._interpolationConfig);
|
---|
120 | if (ast)
|
---|
121 | this._reportExpressionParserErrors(ast.errors, sourceSpan);
|
---|
122 | this._checkPipes(ast, sourceSpan);
|
---|
123 | return ast;
|
---|
124 | }
|
---|
125 | catch (e) {
|
---|
126 | this._reportError("" + e, sourceSpan);
|
---|
127 | return this._exprParser.wrapLiteralPrimitive('ERROR', sourceInfo, absoluteOffset);
|
---|
128 | }
|
---|
129 | };
|
---|
130 | /**
|
---|
131 | * Similar to `parseInterpolation`, but treats the provided string as a single expression
|
---|
132 | * element that would normally appear within the interpolation prefix and suffix (`{{` and `}}`).
|
---|
133 | * This is used for parsing the switch expression in ICUs.
|
---|
134 | */
|
---|
135 | BindingParser.prototype.parseInterpolationExpression = function (expression, sourceSpan) {
|
---|
136 | var sourceInfo = sourceSpan.start.toString();
|
---|
137 | var absoluteOffset = sourceSpan.start.offset;
|
---|
138 | try {
|
---|
139 | var ast = this._exprParser.parseInterpolationExpression(expression, sourceInfo, absoluteOffset);
|
---|
140 | if (ast)
|
---|
141 | this._reportExpressionParserErrors(ast.errors, sourceSpan);
|
---|
142 | this._checkPipes(ast, sourceSpan);
|
---|
143 | return ast;
|
---|
144 | }
|
---|
145 | catch (e) {
|
---|
146 | this._reportError("" + e, sourceSpan);
|
---|
147 | return this._exprParser.wrapLiteralPrimitive('ERROR', sourceInfo, absoluteOffset);
|
---|
148 | }
|
---|
149 | };
|
---|
150 | /**
|
---|
151 | * Parses the bindings in a microsyntax expression, and converts them to
|
---|
152 | * `ParsedProperty` or `ParsedVariable`.
|
---|
153 | *
|
---|
154 | * @param tplKey template binding name
|
---|
155 | * @param tplValue template binding value
|
---|
156 | * @param sourceSpan span of template binding relative to entire the template
|
---|
157 | * @param absoluteValueOffset start of the tplValue relative to the entire template
|
---|
158 | * @param targetMatchableAttrs potential attributes to match in the template
|
---|
159 | * @param targetProps target property bindings in the template
|
---|
160 | * @param targetVars target variables in the template
|
---|
161 | */
|
---|
162 | BindingParser.prototype.parseInlineTemplateBinding = function (tplKey, tplValue, sourceSpan, absoluteValueOffset, targetMatchableAttrs, targetProps, targetVars, isIvyAst) {
|
---|
163 | var e_1, _a;
|
---|
164 | var absoluteKeyOffset = sourceSpan.start.offset + TEMPLATE_ATTR_PREFIX.length;
|
---|
165 | var bindings = this._parseTemplateBindings(tplKey, tplValue, sourceSpan, absoluteKeyOffset, absoluteValueOffset);
|
---|
166 | try {
|
---|
167 | for (var bindings_1 = tslib_1.__values(bindings), bindings_1_1 = bindings_1.next(); !bindings_1_1.done; bindings_1_1 = bindings_1.next()) {
|
---|
168 | var binding = bindings_1_1.value;
|
---|
169 | // sourceSpan is for the entire HTML attribute. bindingSpan is for a particular
|
---|
170 | // binding within the microsyntax expression so it's more narrow than sourceSpan.
|
---|
171 | var bindingSpan = moveParseSourceSpan(sourceSpan, binding.sourceSpan);
|
---|
172 | var key = binding.key.source;
|
---|
173 | var keySpan = moveParseSourceSpan(sourceSpan, binding.key.span);
|
---|
174 | if (binding instanceof ast_1.VariableBinding) {
|
---|
175 | var value = binding.value ? binding.value.source : '$implicit';
|
---|
176 | var valueSpan = binding.value ? moveParseSourceSpan(sourceSpan, binding.value.span) : undefined;
|
---|
177 | targetVars.push(new ast_1.ParsedVariable(key, value, bindingSpan, keySpan, valueSpan));
|
---|
178 | }
|
---|
179 | else if (binding.value) {
|
---|
180 | var srcSpan = isIvyAst ? bindingSpan : sourceSpan;
|
---|
181 | var valueSpan = moveParseSourceSpan(sourceSpan, binding.value.ast.sourceSpan);
|
---|
182 | this._parsePropertyAst(key, binding.value, srcSpan, keySpan, valueSpan, targetMatchableAttrs, targetProps);
|
---|
183 | }
|
---|
184 | else {
|
---|
185 | targetMatchableAttrs.push([key, '' /* value */]);
|
---|
186 | // Since this is a literal attribute with no RHS, source span should be
|
---|
187 | // just the key span.
|
---|
188 | this.parseLiteralAttr(key, null /* value */, keySpan, absoluteValueOffset, undefined /* valueSpan */, targetMatchableAttrs, targetProps, keySpan);
|
---|
189 | }
|
---|
190 | }
|
---|
191 | }
|
---|
192 | catch (e_1_1) { e_1 = { error: e_1_1 }; }
|
---|
193 | finally {
|
---|
194 | try {
|
---|
195 | if (bindings_1_1 && !bindings_1_1.done && (_a = bindings_1.return)) _a.call(bindings_1);
|
---|
196 | }
|
---|
197 | finally { if (e_1) throw e_1.error; }
|
---|
198 | }
|
---|
199 | };
|
---|
200 | /**
|
---|
201 | * Parses the bindings in a microsyntax expression, e.g.
|
---|
202 | * ```
|
---|
203 | * <tag *tplKey="let value1 = prop; let value2 = localVar">
|
---|
204 | * ```
|
---|
205 | *
|
---|
206 | * @param tplKey template binding name
|
---|
207 | * @param tplValue template binding value
|
---|
208 | * @param sourceSpan span of template binding relative to entire the template
|
---|
209 | * @param absoluteKeyOffset start of the `tplKey`
|
---|
210 | * @param absoluteValueOffset start of the `tplValue`
|
---|
211 | */
|
---|
212 | BindingParser.prototype._parseTemplateBindings = function (tplKey, tplValue, sourceSpan, absoluteKeyOffset, absoluteValueOffset) {
|
---|
213 | var _this = this;
|
---|
214 | var sourceInfo = sourceSpan.start.toString();
|
---|
215 | try {
|
---|
216 | var bindingsResult = this._exprParser.parseTemplateBindings(tplKey, tplValue, sourceInfo, absoluteKeyOffset, absoluteValueOffset);
|
---|
217 | this._reportExpressionParserErrors(bindingsResult.errors, sourceSpan);
|
---|
218 | bindingsResult.templateBindings.forEach(function (binding) {
|
---|
219 | if (binding.value instanceof ast_1.ASTWithSource) {
|
---|
220 | _this._checkPipes(binding.value, sourceSpan);
|
---|
221 | }
|
---|
222 | });
|
---|
223 | bindingsResult.warnings.forEach(function (warning) {
|
---|
224 | _this._reportError(warning, sourceSpan, parse_util_1.ParseErrorLevel.WARNING);
|
---|
225 | });
|
---|
226 | return bindingsResult.templateBindings;
|
---|
227 | }
|
---|
228 | catch (e) {
|
---|
229 | this._reportError("" + e, sourceSpan);
|
---|
230 | return [];
|
---|
231 | }
|
---|
232 | };
|
---|
233 | BindingParser.prototype.parseLiteralAttr = function (name, value, sourceSpan, absoluteOffset, valueSpan, targetMatchableAttrs,
|
---|
234 | // TODO(atscott): keySpan is only optional here so VE template parser implementation does not
|
---|
235 | // have to change This should be required when VE is removed.
|
---|
236 | targetProps, keySpan) {
|
---|
237 | if (isAnimationLabel(name)) {
|
---|
238 | name = name.substring(1);
|
---|
239 | if (keySpan !== undefined) {
|
---|
240 | keySpan = moveParseSourceSpan(keySpan, new ast_1.AbsoluteSourceSpan(keySpan.start.offset + 1, keySpan.end.offset));
|
---|
241 | }
|
---|
242 | if (value) {
|
---|
243 | this._reportError("Assigning animation triggers via @prop=\"exp\" attributes with an expression is invalid." +
|
---|
244 | " Use property bindings (e.g. [@prop]=\"exp\") or use an attribute without a value (e.g. @prop) instead.", sourceSpan, parse_util_1.ParseErrorLevel.ERROR);
|
---|
245 | }
|
---|
246 | this._parseAnimation(name, value, sourceSpan, absoluteOffset, keySpan, valueSpan, targetMatchableAttrs, targetProps);
|
---|
247 | }
|
---|
248 | else {
|
---|
249 | targetProps.push(new ast_1.ParsedProperty(name, this._exprParser.wrapLiteralPrimitive(value, '', absoluteOffset), ast_1.ParsedPropertyType.LITERAL_ATTR, sourceSpan, keySpan, valueSpan));
|
---|
250 | }
|
---|
251 | };
|
---|
252 | BindingParser.prototype.parsePropertyBinding = function (name, expression, isHost, sourceSpan, absoluteOffset, valueSpan,
|
---|
253 | // TODO(atscott): keySpan is only optional here so VE template parser implementation does not
|
---|
254 | // have to change This should be required when VE is removed.
|
---|
255 | targetMatchableAttrs, targetProps, keySpan) {
|
---|
256 | if (name.length === 0) {
|
---|
257 | this._reportError("Property name is missing in binding", sourceSpan);
|
---|
258 | }
|
---|
259 | var isAnimationProp = false;
|
---|
260 | if (name.startsWith(ANIMATE_PROP_PREFIX)) {
|
---|
261 | isAnimationProp = true;
|
---|
262 | name = name.substring(ANIMATE_PROP_PREFIX.length);
|
---|
263 | if (keySpan !== undefined) {
|
---|
264 | keySpan = moveParseSourceSpan(keySpan, new ast_1.AbsoluteSourceSpan(keySpan.start.offset + ANIMATE_PROP_PREFIX.length, keySpan.end.offset));
|
---|
265 | }
|
---|
266 | }
|
---|
267 | else if (isAnimationLabel(name)) {
|
---|
268 | isAnimationProp = true;
|
---|
269 | name = name.substring(1);
|
---|
270 | if (keySpan !== undefined) {
|
---|
271 | keySpan = moveParseSourceSpan(keySpan, new ast_1.AbsoluteSourceSpan(keySpan.start.offset + 1, keySpan.end.offset));
|
---|
272 | }
|
---|
273 | }
|
---|
274 | if (isAnimationProp) {
|
---|
275 | this._parseAnimation(name, expression, sourceSpan, absoluteOffset, keySpan, valueSpan, targetMatchableAttrs, targetProps);
|
---|
276 | }
|
---|
277 | else {
|
---|
278 | this._parsePropertyAst(name, this._parseBinding(expression, isHost, valueSpan || sourceSpan, absoluteOffset), sourceSpan, keySpan, valueSpan, targetMatchableAttrs, targetProps);
|
---|
279 | }
|
---|
280 | };
|
---|
281 | BindingParser.prototype.parsePropertyInterpolation = function (name, value, sourceSpan, valueSpan, targetMatchableAttrs,
|
---|
282 | // TODO(atscott): keySpan is only optional here so VE template parser implementation does not
|
---|
283 | // have to change This should be required when VE is removed.
|
---|
284 | targetProps, keySpan) {
|
---|
285 | var expr = this.parseInterpolation(value, valueSpan || sourceSpan);
|
---|
286 | if (expr) {
|
---|
287 | this._parsePropertyAst(name, expr, sourceSpan, keySpan, valueSpan, targetMatchableAttrs, targetProps);
|
---|
288 | return true;
|
---|
289 | }
|
---|
290 | return false;
|
---|
291 | };
|
---|
292 | BindingParser.prototype._parsePropertyAst = function (name, ast, sourceSpan, keySpan, valueSpan, targetMatchableAttrs, targetProps) {
|
---|
293 | targetMatchableAttrs.push([name, ast.source]);
|
---|
294 | targetProps.push(new ast_1.ParsedProperty(name, ast, ast_1.ParsedPropertyType.DEFAULT, sourceSpan, keySpan, valueSpan));
|
---|
295 | };
|
---|
296 | BindingParser.prototype._parseAnimation = function (name, expression, sourceSpan, absoluteOffset, keySpan, valueSpan, targetMatchableAttrs, targetProps) {
|
---|
297 | if (name.length === 0) {
|
---|
298 | this._reportError('Animation trigger is missing', sourceSpan);
|
---|
299 | }
|
---|
300 | // This will occur when a @trigger is not paired with an expression.
|
---|
301 | // For animations it is valid to not have an expression since */void
|
---|
302 | // states will be applied by angular when the element is attached/detached
|
---|
303 | var ast = this._parseBinding(expression || 'undefined', false, valueSpan || sourceSpan, absoluteOffset);
|
---|
304 | targetMatchableAttrs.push([name, ast.source]);
|
---|
305 | targetProps.push(new ast_1.ParsedProperty(name, ast, ast_1.ParsedPropertyType.ANIMATION, sourceSpan, keySpan, valueSpan));
|
---|
306 | };
|
---|
307 | BindingParser.prototype._parseBinding = function (value, isHostBinding, sourceSpan, absoluteOffset) {
|
---|
308 | var sourceInfo = (sourceSpan && sourceSpan.start || '(unknown)').toString();
|
---|
309 | try {
|
---|
310 | var ast = isHostBinding ?
|
---|
311 | this._exprParser.parseSimpleBinding(value, sourceInfo, absoluteOffset, this._interpolationConfig) :
|
---|
312 | this._exprParser.parseBinding(value, sourceInfo, absoluteOffset, this._interpolationConfig);
|
---|
313 | if (ast)
|
---|
314 | this._reportExpressionParserErrors(ast.errors, sourceSpan);
|
---|
315 | this._checkPipes(ast, sourceSpan);
|
---|
316 | return ast;
|
---|
317 | }
|
---|
318 | catch (e) {
|
---|
319 | this._reportError("" + e, sourceSpan);
|
---|
320 | return this._exprParser.wrapLiteralPrimitive('ERROR', sourceInfo, absoluteOffset);
|
---|
321 | }
|
---|
322 | };
|
---|
323 | BindingParser.prototype.createBoundElementProperty = function (elementSelector, boundProp, skipValidation, mapPropertyName) {
|
---|
324 | if (skipValidation === void 0) { skipValidation = false; }
|
---|
325 | if (mapPropertyName === void 0) { mapPropertyName = true; }
|
---|
326 | if (boundProp.isAnimation) {
|
---|
327 | return new ast_1.BoundElementProperty(boundProp.name, 4 /* Animation */, core_1.SecurityContext.NONE, boundProp.expression, null, boundProp.sourceSpan, boundProp.keySpan, boundProp.valueSpan);
|
---|
328 | }
|
---|
329 | var unit = null;
|
---|
330 | var bindingType = undefined;
|
---|
331 | var boundPropertyName = null;
|
---|
332 | var parts = boundProp.name.split(PROPERTY_PARTS_SEPARATOR);
|
---|
333 | var securityContexts = undefined;
|
---|
334 | // Check for special cases (prefix style, attr, class)
|
---|
335 | if (parts.length > 1) {
|
---|
336 | if (parts[0] == ATTRIBUTE_PREFIX) {
|
---|
337 | boundPropertyName = parts.slice(1).join(PROPERTY_PARTS_SEPARATOR);
|
---|
338 | if (!skipValidation) {
|
---|
339 | this._validatePropertyOrAttributeName(boundPropertyName, boundProp.sourceSpan, true);
|
---|
340 | }
|
---|
341 | securityContexts = calcPossibleSecurityContexts(this._schemaRegistry, elementSelector, boundPropertyName, true);
|
---|
342 | var nsSeparatorIdx = boundPropertyName.indexOf(':');
|
---|
343 | if (nsSeparatorIdx > -1) {
|
---|
344 | var ns = boundPropertyName.substring(0, nsSeparatorIdx);
|
---|
345 | var name_1 = boundPropertyName.substring(nsSeparatorIdx + 1);
|
---|
346 | boundPropertyName = tags_1.mergeNsAndName(ns, name_1);
|
---|
347 | }
|
---|
348 | bindingType = 1 /* Attribute */;
|
---|
349 | }
|
---|
350 | else if (parts[0] == CLASS_PREFIX) {
|
---|
351 | boundPropertyName = parts[1];
|
---|
352 | bindingType = 2 /* Class */;
|
---|
353 | securityContexts = [core_1.SecurityContext.NONE];
|
---|
354 | }
|
---|
355 | else if (parts[0] == STYLE_PREFIX) {
|
---|
356 | unit = parts.length > 2 ? parts[2] : null;
|
---|
357 | boundPropertyName = parts[1];
|
---|
358 | bindingType = 3 /* Style */;
|
---|
359 | securityContexts = [core_1.SecurityContext.STYLE];
|
---|
360 | }
|
---|
361 | }
|
---|
362 | // If not a special case, use the full property name
|
---|
363 | if (boundPropertyName === null) {
|
---|
364 | var mappedPropName = this._schemaRegistry.getMappedPropName(boundProp.name);
|
---|
365 | boundPropertyName = mapPropertyName ? mappedPropName : boundProp.name;
|
---|
366 | securityContexts = calcPossibleSecurityContexts(this._schemaRegistry, elementSelector, mappedPropName, false);
|
---|
367 | bindingType = 0 /* Property */;
|
---|
368 | if (!skipValidation) {
|
---|
369 | this._validatePropertyOrAttributeName(mappedPropName, boundProp.sourceSpan, false);
|
---|
370 | }
|
---|
371 | }
|
---|
372 | return new ast_1.BoundElementProperty(boundPropertyName, bindingType, securityContexts[0], boundProp.expression, unit, boundProp.sourceSpan, boundProp.keySpan, boundProp.valueSpan);
|
---|
373 | };
|
---|
374 | // TODO: keySpan should be required but was made optional to avoid changing VE parser.
|
---|
375 | BindingParser.prototype.parseEvent = function (name, expression, sourceSpan, handlerSpan, targetMatchableAttrs, targetEvents, keySpan) {
|
---|
376 | if (name.length === 0) {
|
---|
377 | this._reportError("Event name is missing in binding", sourceSpan);
|
---|
378 | }
|
---|
379 | if (isAnimationLabel(name)) {
|
---|
380 | name = name.substr(1);
|
---|
381 | if (keySpan !== undefined) {
|
---|
382 | keySpan = moveParseSourceSpan(keySpan, new ast_1.AbsoluteSourceSpan(keySpan.start.offset + 1, keySpan.end.offset));
|
---|
383 | }
|
---|
384 | this._parseAnimationEvent(name, expression, sourceSpan, handlerSpan, targetEvents, keySpan);
|
---|
385 | }
|
---|
386 | else {
|
---|
387 | this._parseRegularEvent(name, expression, sourceSpan, handlerSpan, targetMatchableAttrs, targetEvents, keySpan);
|
---|
388 | }
|
---|
389 | };
|
---|
390 | BindingParser.prototype.calcPossibleSecurityContexts = function (selector, propName, isAttribute) {
|
---|
391 | var prop = this._schemaRegistry.getMappedPropName(propName);
|
---|
392 | return calcPossibleSecurityContexts(this._schemaRegistry, selector, prop, isAttribute);
|
---|
393 | };
|
---|
394 | BindingParser.prototype._parseAnimationEvent = function (name, expression, sourceSpan, handlerSpan, targetEvents, keySpan) {
|
---|
395 | var matches = util_1.splitAtPeriod(name, [name, '']);
|
---|
396 | var eventName = matches[0];
|
---|
397 | var phase = matches[1].toLowerCase();
|
---|
398 | var ast = this._parseAction(expression, handlerSpan);
|
---|
399 | targetEvents.push(new ast_1.ParsedEvent(eventName, phase, 1 /* Animation */, ast, sourceSpan, handlerSpan, keySpan));
|
---|
400 | if (eventName.length === 0) {
|
---|
401 | this._reportError("Animation event name is missing in binding", sourceSpan);
|
---|
402 | }
|
---|
403 | if (phase) {
|
---|
404 | if (phase !== 'start' && phase !== 'done') {
|
---|
405 | this._reportError("The provided animation output phase value \"" + phase + "\" for \"@" + eventName + "\" is not supported (use start or done)", sourceSpan);
|
---|
406 | }
|
---|
407 | }
|
---|
408 | else {
|
---|
409 | this._reportError("The animation trigger output event (@" + eventName + ") is missing its phase value name (start or done are currently supported)", sourceSpan);
|
---|
410 | }
|
---|
411 | };
|
---|
412 | BindingParser.prototype._parseRegularEvent = function (name, expression, sourceSpan, handlerSpan, targetMatchableAttrs, targetEvents, keySpan) {
|
---|
413 | // long format: 'target: eventName'
|
---|
414 | var _a = tslib_1.__read(util_1.splitAtColon(name, [null, name]), 2), target = _a[0], eventName = _a[1];
|
---|
415 | var ast = this._parseAction(expression, handlerSpan);
|
---|
416 | targetMatchableAttrs.push([name, ast.source]);
|
---|
417 | targetEvents.push(new ast_1.ParsedEvent(eventName, target, 0 /* Regular */, ast, sourceSpan, handlerSpan, keySpan));
|
---|
418 | // Don't detect directives for event names for now,
|
---|
419 | // so don't add the event name to the matchableAttrs
|
---|
420 | };
|
---|
421 | BindingParser.prototype._parseAction = function (value, sourceSpan) {
|
---|
422 | var sourceInfo = (sourceSpan && sourceSpan.start || '(unknown').toString();
|
---|
423 | var absoluteOffset = (sourceSpan && sourceSpan.start) ? sourceSpan.start.offset : 0;
|
---|
424 | try {
|
---|
425 | var ast = this._exprParser.parseAction(value, sourceInfo, absoluteOffset, this._interpolationConfig);
|
---|
426 | if (ast) {
|
---|
427 | this._reportExpressionParserErrors(ast.errors, sourceSpan);
|
---|
428 | }
|
---|
429 | if (!ast || ast.ast instanceof ast_1.EmptyExpr) {
|
---|
430 | this._reportError("Empty expressions are not allowed", sourceSpan);
|
---|
431 | return this._exprParser.wrapLiteralPrimitive('ERROR', sourceInfo, absoluteOffset);
|
---|
432 | }
|
---|
433 | this._checkPipes(ast, sourceSpan);
|
---|
434 | return ast;
|
---|
435 | }
|
---|
436 | catch (e) {
|
---|
437 | this._reportError("" + e, sourceSpan);
|
---|
438 | return this._exprParser.wrapLiteralPrimitive('ERROR', sourceInfo, absoluteOffset);
|
---|
439 | }
|
---|
440 | };
|
---|
441 | BindingParser.prototype._reportError = function (message, sourceSpan, level) {
|
---|
442 | if (level === void 0) { level = parse_util_1.ParseErrorLevel.ERROR; }
|
---|
443 | this.errors.push(new parse_util_1.ParseError(sourceSpan, message, level));
|
---|
444 | };
|
---|
445 | BindingParser.prototype._reportExpressionParserErrors = function (errors, sourceSpan) {
|
---|
446 | var e_2, _a;
|
---|
447 | try {
|
---|
448 | for (var errors_1 = tslib_1.__values(errors), errors_1_1 = errors_1.next(); !errors_1_1.done; errors_1_1 = errors_1.next()) {
|
---|
449 | var error = errors_1_1.value;
|
---|
450 | this._reportError(error.message, sourceSpan);
|
---|
451 | }
|
---|
452 | }
|
---|
453 | catch (e_2_1) { e_2 = { error: e_2_1 }; }
|
---|
454 | finally {
|
---|
455 | try {
|
---|
456 | if (errors_1_1 && !errors_1_1.done && (_a = errors_1.return)) _a.call(errors_1);
|
---|
457 | }
|
---|
458 | finally { if (e_2) throw e_2.error; }
|
---|
459 | }
|
---|
460 | };
|
---|
461 | // Make sure all the used pipes are known in `this.pipesByName`
|
---|
462 | BindingParser.prototype._checkPipes = function (ast, sourceSpan) {
|
---|
463 | var _this = this;
|
---|
464 | if (ast && this.pipesByName) {
|
---|
465 | var collector = new PipeCollector();
|
---|
466 | ast.visit(collector);
|
---|
467 | collector.pipes.forEach(function (ast, pipeName) {
|
---|
468 | var pipeMeta = _this.pipesByName.get(pipeName);
|
---|
469 | if (!pipeMeta) {
|
---|
470 | _this._reportError("The pipe '" + pipeName + "' could not be found", new parse_util_1.ParseSourceSpan(sourceSpan.start.moveBy(ast.span.start), sourceSpan.start.moveBy(ast.span.end)));
|
---|
471 | }
|
---|
472 | else {
|
---|
473 | _this._usedPipes.set(pipeName, pipeMeta);
|
---|
474 | }
|
---|
475 | });
|
---|
476 | }
|
---|
477 | };
|
---|
478 | /**
|
---|
479 | * @param propName the name of the property / attribute
|
---|
480 | * @param sourceSpan
|
---|
481 | * @param isAttr true when binding to an attribute
|
---|
482 | */
|
---|
483 | BindingParser.prototype._validatePropertyOrAttributeName = function (propName, sourceSpan, isAttr) {
|
---|
484 | var report = isAttr ? this._schemaRegistry.validateAttribute(propName) :
|
---|
485 | this._schemaRegistry.validateProperty(propName);
|
---|
486 | if (report.error) {
|
---|
487 | this._reportError(report.msg, sourceSpan, parse_util_1.ParseErrorLevel.ERROR);
|
---|
488 | }
|
---|
489 | };
|
---|
490 | return BindingParser;
|
---|
491 | }());
|
---|
492 | exports.BindingParser = BindingParser;
|
---|
493 | var PipeCollector = /** @class */ (function (_super) {
|
---|
494 | tslib_1.__extends(PipeCollector, _super);
|
---|
495 | function PipeCollector() {
|
---|
496 | var _this = _super !== null && _super.apply(this, arguments) || this;
|
---|
497 | _this.pipes = new Map();
|
---|
498 | return _this;
|
---|
499 | }
|
---|
500 | PipeCollector.prototype.visitPipe = function (ast, context) {
|
---|
501 | this.pipes.set(ast.name, ast);
|
---|
502 | ast.exp.visit(this);
|
---|
503 | this.visitAll(ast.args, context);
|
---|
504 | return null;
|
---|
505 | };
|
---|
506 | return PipeCollector;
|
---|
507 | }(ast_1.RecursiveAstVisitor));
|
---|
508 | exports.PipeCollector = PipeCollector;
|
---|
509 | function isAnimationLabel(name) {
|
---|
510 | return name[0] == '@';
|
---|
511 | }
|
---|
512 | function calcPossibleSecurityContexts(registry, selector, propName, isAttribute) {
|
---|
513 | var ctxs = [];
|
---|
514 | selector_1.CssSelector.parse(selector).forEach(function (selector) {
|
---|
515 | var elementNames = selector.element ? [selector.element] : registry.allKnownElementNames();
|
---|
516 | var notElementNames = new Set(selector.notSelectors.filter(function (selector) { return selector.isElementSelector(); })
|
---|
517 | .map(function (selector) { return selector.element; }));
|
---|
518 | var possibleElementNames = elementNames.filter(function (elementName) { return !notElementNames.has(elementName); });
|
---|
519 | ctxs.push.apply(ctxs, tslib_1.__spreadArray([], tslib_1.__read(possibleElementNames.map(function (elementName) { return registry.securityContext(elementName, propName, isAttribute); }))));
|
---|
520 | });
|
---|
521 | return ctxs.length === 0 ? [core_1.SecurityContext.NONE] : Array.from(new Set(ctxs)).sort();
|
---|
522 | }
|
---|
523 | exports.calcPossibleSecurityContexts = calcPossibleSecurityContexts;
|
---|
524 | /**
|
---|
525 | * Compute a new ParseSourceSpan based off an original `sourceSpan` by using
|
---|
526 | * absolute offsets from the specified `absoluteSpan`.
|
---|
527 | *
|
---|
528 | * @param sourceSpan original source span
|
---|
529 | * @param absoluteSpan absolute source span to move to
|
---|
530 | */
|
---|
531 | function moveParseSourceSpan(sourceSpan, absoluteSpan) {
|
---|
532 | // The difference of two absolute offsets provide the relative offset
|
---|
533 | var startDiff = absoluteSpan.start - sourceSpan.start.offset;
|
---|
534 | var endDiff = absoluteSpan.end - sourceSpan.end.offset;
|
---|
535 | return new parse_util_1.ParseSourceSpan(sourceSpan.start.moveBy(startDiff), sourceSpan.end.moveBy(endDiff), sourceSpan.fullStart.moveBy(startDiff), sourceSpan.details);
|
---|
536 | }
|
---|
537 | });
|
---|
538 | //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"binding_parser.js","sourceRoot":"","sources":["../../../../../../../packages/compiler/src/template_parser/binding_parser.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\n\nimport {CompileDirectiveSummary, CompilePipeSummary} from '../compile_metadata';\nimport {SecurityContext} from '../core';\nimport {AbsoluteSourceSpan, ASTWithSource, BindingPipe, BindingType, BoundElementProperty, EmptyExpr, ParsedEvent, ParsedEventType, ParsedProperty, ParsedPropertyType, ParsedVariable, ParserError, RecursiveAstVisitor, TemplateBinding, VariableBinding} from '../expression_parser/ast';\nimport {Parser} from '../expression_parser/parser';\nimport {InterpolationConfig} from '../ml_parser/interpolation_config';\nimport {mergeNsAndName} from '../ml_parser/tags';\nimport {ParseError, ParseErrorLevel, ParseLocation, ParseSourceSpan} from '../parse_util';\nimport {ElementSchemaRegistry} from '../schema/element_schema_registry';\nimport {CssSelector} from '../selector';\nimport {splitAtColon, splitAtPeriod} from '../util';\n\nconst PROPERTY_PARTS_SEPARATOR = '.';\nconst ATTRIBUTE_PREFIX = 'attr';\nconst CLASS_PREFIX = 'class';\nconst STYLE_PREFIX = 'style';\nconst TEMPLATE_ATTR_PREFIX = '*';\nconst ANIMATE_PROP_PREFIX = 'animate-';\n\n/**\n * Parses bindings in templates and in the directive host area.\n */\nexport class BindingParser {\n  pipesByName: Map<string, CompilePipeSummary>|null = null;\n\n  private _usedPipes: Map<string, CompilePipeSummary> = new Map();\n\n  constructor(\n      private _exprParser: Parser, private _interpolationConfig: InterpolationConfig,\n      private _schemaRegistry: ElementSchemaRegistry, pipes: CompilePipeSummary[]|null,\n      public errors: ParseError[]) {\n    // When the `pipes` parameter is `null`, do not check for used pipes\n    // This is used in IVY when we might not know the available pipes at compile time\n    if (pipes) {\n      const pipesByName: Map<string, CompilePipeSummary> = new Map();\n      pipes.forEach(pipe => pipesByName.set(pipe.name, pipe));\n      this.pipesByName = pipesByName;\n    }\n  }\n\n  get interpolationConfig(): InterpolationConfig {\n    return this._interpolationConfig;\n  }\n\n  getUsedPipes(): CompilePipeSummary[] {\n    return Array.from(this._usedPipes.values());\n  }\n\n  createBoundHostProperties(dirMeta: CompileDirectiveSummary, sourceSpan: ParseSourceSpan):\n      ParsedProperty[]|null {\n    if (dirMeta.hostProperties) {\n      const boundProps: ParsedProperty[] = [];\n      Object.keys(dirMeta.hostProperties).forEach(propName => {\n        const expression = dirMeta.hostProperties[propName];\n        if (typeof expression === 'string') {\n          this.parsePropertyBinding(\n              propName, expression, true, sourceSpan, sourceSpan.start.offset, undefined, [],\n              // Use the `sourceSpan` for  `keySpan`. This isn't really accurate, but neither is the\n              // sourceSpan, as it represents the sourceSpan of the host itself rather than the\n              // source of the host binding (which doesn't exist in the template). Regardless,\n              // neither of these values are used in Ivy but are only here to satisfy the function\n              // signature. This should likely be refactored in the future so that `sourceSpan`\n              // isn't being used inaccurately.\n              boundProps, sourceSpan);\n        } else {\n          this._reportError(\n              `Value of the host property binding \"${\n                  propName}\" needs to be a string representing an expression but got \"${\n                  expression}\" (${typeof expression})`,\n              sourceSpan);\n        }\n      });\n      return boundProps;\n    }\n    return null;\n  }\n\n  createDirectiveHostPropertyAsts(\n      dirMeta: CompileDirectiveSummary, elementSelector: string,\n      sourceSpan: ParseSourceSpan): BoundElementProperty[]|null {\n    const boundProps = this.createBoundHostProperties(dirMeta, sourceSpan);\n    return boundProps &&\n        boundProps.map((prop) => this.createBoundElementProperty(elementSelector, prop));\n  }\n\n  createDirectiveHostEventAsts(dirMeta: CompileDirectiveSummary, sourceSpan: ParseSourceSpan):\n      ParsedEvent[]|null {\n    if (dirMeta.hostListeners) {\n      const targetEvents: ParsedEvent[] = [];\n      Object.keys(dirMeta.hostListeners).forEach(propName => {\n        const expression = dirMeta.hostListeners[propName];\n        if (typeof expression === 'string') {\n          // Use the `sourceSpan` for  `keySpan` and `handlerSpan`. This isn't really accurate, but\n          // neither is the `sourceSpan`, as it represents the `sourceSpan` of the host itself\n          // rather than the source of the host binding (which doesn't exist in the template).\n          // Regardless, neither of these values are used in Ivy but are only here to satisfy the\n          // function signature. This should likely be refactored in the future so that `sourceSpan`\n          // isn't being used inaccurately.\n          this.parseEvent(\n              propName, expression, sourceSpan, sourceSpan, [], targetEvents, sourceSpan);\n        } else {\n          this._reportError(\n              `Value of the host listener \"${\n                  propName}\" needs to be a string representing an expression but got \"${\n                  expression}\" (${typeof expression})`,\n              sourceSpan);\n        }\n      });\n      return targetEvents;\n    }\n    return null;\n  }\n\n  parseInterpolation(value: string, sourceSpan: ParseSourceSpan): ASTWithSource {\n    const sourceInfo = sourceSpan.start.toString();\n    const absoluteOffset = sourceSpan.fullStart.offset;\n\n    try {\n      const ast = this._exprParser.parseInterpolation(\n          value, sourceInfo, absoluteOffset, this._interpolationConfig)!;\n      if (ast) this._reportExpressionParserErrors(ast.errors, sourceSpan);\n      this._checkPipes(ast, sourceSpan);\n      return ast;\n    } catch (e) {\n      this._reportError(`${e}`, sourceSpan);\n      return this._exprParser.wrapLiteralPrimitive('ERROR', sourceInfo, absoluteOffset);\n    }\n  }\n\n  /**\n   * Similar to `parseInterpolation`, but treats the provided string as a single expression\n   * element that would normally appear within the interpolation prefix and suffix (`{{` and `}}`).\n   * This is used for parsing the switch expression in ICUs.\n   */\n  parseInterpolationExpression(expression: string, sourceSpan: ParseSourceSpan): ASTWithSource {\n    const sourceInfo = sourceSpan.start.toString();\n    const absoluteOffset = sourceSpan.start.offset;\n\n    try {\n      const ast =\n          this._exprParser.parseInterpolationExpression(expression, sourceInfo, absoluteOffset);\n      if (ast) this._reportExpressionParserErrors(ast.errors, sourceSpan);\n      this._checkPipes(ast, sourceSpan);\n      return ast;\n    } catch (e) {\n      this._reportError(`${e}`, sourceSpan);\n      return this._exprParser.wrapLiteralPrimitive('ERROR', sourceInfo, absoluteOffset);\n    }\n  }\n\n  /**\n   * Parses the bindings in a microsyntax expression, and converts them to\n   * `ParsedProperty` or `ParsedVariable`.\n   *\n   * @param tplKey template binding name\n   * @param tplValue template binding value\n   * @param sourceSpan span of template binding relative to entire the template\n   * @param absoluteValueOffset start of the tplValue relative to the entire template\n   * @param targetMatchableAttrs potential attributes to match in the template\n   * @param targetProps target property bindings in the template\n   * @param targetVars target variables in the template\n   */\n  parseInlineTemplateBinding(\n      tplKey: string, tplValue: string, sourceSpan: ParseSourceSpan, absoluteValueOffset: number,\n      targetMatchableAttrs: string[][], targetProps: ParsedProperty[], targetVars: ParsedVariable[],\n      isIvyAst: boolean) {\n    const absoluteKeyOffset = sourceSpan.start.offset + TEMPLATE_ATTR_PREFIX.length;\n    const bindings = this._parseTemplateBindings(\n        tplKey, tplValue, sourceSpan, absoluteKeyOffset, absoluteValueOffset);\n\n    for (const binding of bindings) {\n      // sourceSpan is for the entire HTML attribute. bindingSpan is for a particular\n      // binding within the microsyntax expression so it's more narrow than sourceSpan.\n      const bindingSpan = moveParseSourceSpan(sourceSpan, binding.sourceSpan);\n      const key = binding.key.source;\n      const keySpan = moveParseSourceSpan(sourceSpan, binding.key.span);\n      if (binding instanceof VariableBinding) {\n        const value = binding.value ? binding.value.source : '$implicit';\n        const valueSpan =\n            binding.value ? moveParseSourceSpan(sourceSpan, binding.value.span) : undefined;\n        targetVars.push(new ParsedVariable(key, value, bindingSpan, keySpan, valueSpan));\n      } else if (binding.value) {\n        const srcSpan = isIvyAst ? bindingSpan : sourceSpan;\n        const valueSpan = moveParseSourceSpan(sourceSpan, binding.value.ast.sourceSpan);\n        this._parsePropertyAst(\n            key, binding.value, srcSpan, keySpan, valueSpan, targetMatchableAttrs, targetProps);\n      } else {\n        targetMatchableAttrs.push([key, '' /* value */]);\n        // Since this is a literal attribute with no RHS, source span should be\n        // just the key span.\n        this.parseLiteralAttr(\n            key, null /* value */, keySpan, absoluteValueOffset, undefined /* valueSpan */,\n            targetMatchableAttrs, targetProps, keySpan);\n      }\n    }\n  }\n\n  /**\n   * Parses the bindings in a microsyntax expression, e.g.\n   * ```\n   *    <tag *tplKey=\"let value1 = prop; let value2 = localVar\">\n   * ```\n   *\n   * @param tplKey template binding name\n   * @param tplValue template binding value\n   * @param sourceSpan span of template binding relative to entire the template\n   * @param absoluteKeyOffset start of the `tplKey`\n   * @param absoluteValueOffset start of the `tplValue`\n   */\n  private _parseTemplateBindings(\n      tplKey: string, tplValue: string, sourceSpan: ParseSourceSpan, absoluteKeyOffset: number,\n      absoluteValueOffset: number): TemplateBinding[] {\n    const sourceInfo = sourceSpan.start.toString();\n\n    try {\n      const bindingsResult = this._exprParser.parseTemplateBindings(\n          tplKey, tplValue, sourceInfo, absoluteKeyOffset, absoluteValueOffset);\n      this._reportExpressionParserErrors(bindingsResult.errors, sourceSpan);\n      bindingsResult.templateBindings.forEach((binding) => {\n        if (binding.value instanceof ASTWithSource) {\n          this._checkPipes(binding.value, sourceSpan);\n        }\n      });\n      bindingsResult.warnings.forEach((warning) => {\n        this._reportError(warning, sourceSpan, ParseErrorLevel.WARNING);\n      });\n      return bindingsResult.templateBindings;\n    } catch (e) {\n      this._reportError(`${e}`, sourceSpan);\n      return [];\n    }\n  }\n\n  parseLiteralAttr(\n      name: string, value: string|null, sourceSpan: ParseSourceSpan, absoluteOffset: number,\n      valueSpan: ParseSourceSpan|undefined, targetMatchableAttrs: string[][],\n      // TODO(atscott): keySpan is only optional here so VE template parser implementation does not\n      // have to change This should be required when VE is removed.\n      targetProps: ParsedProperty[], keySpan?: ParseSourceSpan) {\n    if (isAnimationLabel(name)) {\n      name = name.substring(1);\n      if (keySpan !== undefined) {\n        keySpan = moveParseSourceSpan(\n            keySpan, new AbsoluteSourceSpan(keySpan.start.offset + 1, keySpan.end.offset));\n      }\n      if (value) {\n        this._reportError(\n            `Assigning animation triggers via @prop=\"exp\" attributes with an expression is invalid.` +\n                ` Use property bindings (e.g. [@prop]=\"exp\") or use an attribute without a value (e.g. @prop) instead.`,\n            sourceSpan, ParseErrorLevel.ERROR);\n      }\n      this._parseAnimation(\n          name, value, sourceSpan, absoluteOffset, keySpan, valueSpan, targetMatchableAttrs,\n          targetProps);\n    } else {\n      targetProps.push(new ParsedProperty(\n          name, this._exprParser.wrapLiteralPrimitive(value, '', absoluteOffset),\n          ParsedPropertyType.LITERAL_ATTR, sourceSpan, keySpan, valueSpan));\n    }\n  }\n\n  parsePropertyBinding(\n      name: string, expression: string, isHost: boolean, sourceSpan: ParseSourceSpan,\n      absoluteOffset: number, valueSpan: ParseSourceSpan|undefined,\n      // TODO(atscott): keySpan is only optional here so VE template parser implementation does not\n      // have to change This should be required when VE is removed.\n      targetMatchableAttrs: string[][], targetProps: ParsedProperty[], keySpan?: ParseSourceSpan) {\n    if (name.length === 0) {\n      this._reportError(`Property name is missing in binding`, sourceSpan);\n    }\n\n    let isAnimationProp = false;\n    if (name.startsWith(ANIMATE_PROP_PREFIX)) {\n      isAnimationProp = true;\n      name = name.substring(ANIMATE_PROP_PREFIX.length);\n      if (keySpan !== undefined) {\n        keySpan = moveParseSourceSpan(\n            keySpan,\n            new AbsoluteSourceSpan(\n                keySpan.start.offset + ANIMATE_PROP_PREFIX.length, keySpan.end.offset));\n      }\n    } else if (isAnimationLabel(name)) {\n      isAnimationProp = true;\n      name = name.substring(1);\n      if (keySpan !== undefined) {\n        keySpan = moveParseSourceSpan(\n            keySpan, new AbsoluteSourceSpan(keySpan.start.offset + 1, keySpan.end.offset));\n      }\n    }\n\n    if (isAnimationProp) {\n      this._parseAnimation(\n          name, expression, sourceSpan, absoluteOffset, keySpan, valueSpan, targetMatchableAttrs,\n          targetProps);\n    } else {\n      this._parsePropertyAst(\n          name, this._parseBinding(expression, isHost, valueSpan || sourceSpan, absoluteOffset),\n          sourceSpan, keySpan, valueSpan, targetMatchableAttrs, targetProps);\n    }\n  }\n\n  parsePropertyInterpolation(\n      name: string, value: string, sourceSpan: ParseSourceSpan,\n      valueSpan: ParseSourceSpan|undefined, targetMatchableAttrs: string[][],\n      // TODO(atscott): keySpan is only optional here so VE template parser implementation does not\n      // have to change This should be required when VE is removed.\n      targetProps: ParsedProperty[], keySpan?: ParseSourceSpan): boolean {\n    const expr = this.parseInterpolation(value, valueSpan || sourceSpan);\n    if (expr) {\n      this._parsePropertyAst(\n          name, expr, sourceSpan, keySpan, valueSpan, targetMatchableAttrs, targetProps);\n      return true;\n    }\n    return false;\n  }\n\n  private _parsePropertyAst(\n      name: string, ast: ASTWithSource, sourceSpan: ParseSourceSpan,\n      keySpan: ParseSourceSpan|undefined, valueSpan: ParseSourceSpan|undefined,\n      targetMatchableAttrs: string[][], targetProps: ParsedProperty[]) {\n    targetMatchableAttrs.push([name, ast.source!]);\n    targetProps.push(\n        new ParsedProperty(name, ast, ParsedPropertyType.DEFAULT, sourceSpan, keySpan, valueSpan));\n  }\n\n  private _parseAnimation(\n      name: string, expression: string|null, sourceSpan: ParseSourceSpan, absoluteOffset: number,\n      keySpan: ParseSourceSpan|undefined, valueSpan: ParseSourceSpan|undefined,\n      targetMatchableAttrs: string[][], targetProps: ParsedProperty[]) {\n    if (name.length === 0) {\n      this._reportError('Animation trigger is missing', sourceSpan);\n    }\n\n    // This will occur when a @trigger is not paired with an expression.\n    // For animations it is valid to not have an expression since */void\n    // states will be applied by angular when the element is attached/detached\n    const ast = this._parseBinding(\n        expression || 'undefined', false, valueSpan || sourceSpan, absoluteOffset);\n    targetMatchableAttrs.push([name, ast.source!]);\n    targetProps.push(new ParsedProperty(\n        name, ast, ParsedPropertyType.ANIMATION, sourceSpan, keySpan, valueSpan));\n  }\n\n  private _parseBinding(\n      value: string, isHostBinding: boolean, sourceSpan: ParseSourceSpan,\n      absoluteOffset: number): ASTWithSource {\n    const sourceInfo = (sourceSpan && sourceSpan.start || '(unknown)').toString();\n\n    try {\n      const ast = isHostBinding ?\n          this._exprParser.parseSimpleBinding(\n              value, sourceInfo, absoluteOffset, this._interpolationConfig) :\n          this._exprParser.parseBinding(\n              value, sourceInfo, absoluteOffset, this._interpolationConfig);\n      if (ast) this._reportExpressionParserErrors(ast.errors, sourceSpan);\n      this._checkPipes(ast, sourceSpan);\n      return ast;\n    } catch (e) {\n      this._reportError(`${e}`, sourceSpan);\n      return this._exprParser.wrapLiteralPrimitive('ERROR', sourceInfo, absoluteOffset);\n    }\n  }\n\n  createBoundElementProperty(\n      elementSelector: string, boundProp: ParsedProperty, skipValidation: boolean = false,\n      mapPropertyName: boolean = true): BoundElementProperty {\n    if (boundProp.isAnimation) {\n      return new BoundElementProperty(\n          boundProp.name, BindingType.Animation, SecurityContext.NONE, boundProp.expression, null,\n          boundProp.sourceSpan, boundProp.keySpan, boundProp.valueSpan);\n    }\n\n    let unit: string|null = null;\n    let bindingType: BindingType = undefined!;\n    let boundPropertyName: string|null = null;\n    const parts = boundProp.name.split(PROPERTY_PARTS_SEPARATOR);\n    let securityContexts: SecurityContext[] = undefined!;\n\n    // Check for special cases (prefix style, attr, class)\n    if (parts.length > 1) {\n      if (parts[0] == ATTRIBUTE_PREFIX) {\n        boundPropertyName = parts.slice(1).join(PROPERTY_PARTS_SEPARATOR);\n        if (!skipValidation) {\n          this._validatePropertyOrAttributeName(boundPropertyName, boundProp.sourceSpan, true);\n        }\n        securityContexts = calcPossibleSecurityContexts(\n            this._schemaRegistry, elementSelector, boundPropertyName, true);\n\n        const nsSeparatorIdx = boundPropertyName.indexOf(':');\n        if (nsSeparatorIdx > -1) {\n          const ns = boundPropertyName.substring(0, nsSeparatorIdx);\n          const name = boundPropertyName.substring(nsSeparatorIdx + 1);\n          boundPropertyName = mergeNsAndName(ns, name);\n        }\n\n        bindingType = BindingType.Attribute;\n      } else if (parts[0] == CLASS_PREFIX) {\n        boundPropertyName = parts[1];\n        bindingType = BindingType.Class;\n        securityContexts = [SecurityContext.NONE];\n      } else if (parts[0] == STYLE_PREFIX) {\n        unit = parts.length > 2 ? parts[2] : null;\n        boundPropertyName = parts[1];\n        bindingType = BindingType.Style;\n        securityContexts = [SecurityContext.STYLE];\n      }\n    }\n\n    // If not a special case, use the full property name\n    if (boundPropertyName === null) {\n      const mappedPropName = this._schemaRegistry.getMappedPropName(boundProp.name);\n      boundPropertyName = mapPropertyName ? mappedPropName : boundProp.name;\n      securityContexts = calcPossibleSecurityContexts(\n          this._schemaRegistry, elementSelector, mappedPropName, false);\n      bindingType = BindingType.Property;\n      if (!skipValidation) {\n        this._validatePropertyOrAttributeName(mappedPropName, boundProp.sourceSpan, false);\n      }\n    }\n\n    return new BoundElementProperty(\n        boundPropertyName, bindingType, securityContexts[0], boundProp.expression, unit,\n        boundProp.sourceSpan, boundProp.keySpan, boundProp.valueSpan);\n  }\n\n  // TODO: keySpan should be required but was made optional to avoid changing VE parser.\n  parseEvent(\n      name: string, expression: string, sourceSpan: ParseSourceSpan, handlerSpan: ParseSourceSpan,\n      targetMatchableAttrs: string[][], targetEvents: ParsedEvent[], keySpan?: ParseSourceSpan) {\n    if (name.length === 0) {\n      this._reportError(`Event name is missing in binding`, sourceSpan);\n    }\n\n    if (isAnimationLabel(name)) {\n      name = name.substr(1);\n      if (keySpan !== undefined) {\n        keySpan = moveParseSourceSpan(\n            keySpan, new AbsoluteSourceSpan(keySpan.start.offset + 1, keySpan.end.offset));\n      }\n      this._parseAnimationEvent(name, expression, sourceSpan, handlerSpan, targetEvents, keySpan);\n    } else {\n      this._parseRegularEvent(\n          name, expression, sourceSpan, handlerSpan, targetMatchableAttrs, targetEvents, keySpan);\n    }\n  }\n\n  calcPossibleSecurityContexts(selector: string, propName: string, isAttribute: boolean):\n      SecurityContext[] {\n    const prop = this._schemaRegistry.getMappedPropName(propName);\n    return calcPossibleSecurityContexts(this._schemaRegistry, selector, prop, isAttribute);\n  }\n\n  private _parseAnimationEvent(\n      name: string, expression: string, sourceSpan: ParseSourceSpan, handlerSpan: ParseSourceSpan,\n      targetEvents: ParsedEvent[], keySpan?: ParseSourceSpan) {\n    const matches = splitAtPeriod(name, [name, '']);\n    const eventName = matches[0];\n    const phase = matches[1].toLowerCase();\n    const ast = this._parseAction(expression, handlerSpan);\n    targetEvents.push(new ParsedEvent(\n        eventName, phase, ParsedEventType.Animation, ast, sourceSpan, handlerSpan, keySpan));\n\n    if (eventName.length === 0) {\n      this._reportError(`Animation event name is missing in binding`, sourceSpan);\n    }\n    if (phase) {\n      if (phase !== 'start' && phase !== 'done') {\n        this._reportError(\n            `The provided animation output phase value \"${phase}\" for \"@${\n                eventName}\" is not supported (use start or done)`,\n            sourceSpan);\n      }\n    } else {\n      this._reportError(\n          `The animation trigger output event (@${\n              eventName}) is missing its phase value name (start or done are currently supported)`,\n          sourceSpan);\n    }\n  }\n\n  private _parseRegularEvent(\n      name: string, expression: string, sourceSpan: ParseSourceSpan, handlerSpan: ParseSourceSpan,\n      targetMatchableAttrs: string[][], targetEvents: ParsedEvent[], keySpan?: ParseSourceSpan) {\n    // long format: 'target: eventName'\n    const [target, eventName] = splitAtColon(name, [null!, name]);\n    const ast = this._parseAction(expression, handlerSpan);\n    targetMatchableAttrs.push([name!, ast.source!]);\n    targetEvents.push(new ParsedEvent(\n        eventName, target, ParsedEventType.Regular, ast, sourceSpan, handlerSpan, keySpan));\n    // Don't detect directives for event names for now,\n    // so don't add the event name to the matchableAttrs\n  }\n\n  private _parseAction(value: string, sourceSpan: ParseSourceSpan): ASTWithSource {\n    const sourceInfo = (sourceSpan && sourceSpan.start || '(unknown').toString();\n    const absoluteOffset = (sourceSpan && sourceSpan.start) ? sourceSpan.start.offset : 0;\n\n    try {\n      const ast = this._exprParser.parseAction(\n          value, sourceInfo, absoluteOffset, this._interpolationConfig);\n      if (ast) {\n        this._reportExpressionParserErrors(ast.errors, sourceSpan);\n      }\n      if (!ast || ast.ast instanceof EmptyExpr) {\n        this._reportError(`Empty expressions are not allowed`, sourceSpan);\n        return this._exprParser.wrapLiteralPrimitive('ERROR', sourceInfo, absoluteOffset);\n      }\n      this._checkPipes(ast, sourceSpan);\n      return ast;\n    } catch (e) {\n      this._reportError(`${e}`, sourceSpan);\n      return this._exprParser.wrapLiteralPrimitive('ERROR', sourceInfo, absoluteOffset);\n    }\n  }\n\n  private _reportError(\n      message: string, sourceSpan: ParseSourceSpan,\n      level: ParseErrorLevel = ParseErrorLevel.ERROR) {\n    this.errors.push(new ParseError(sourceSpan, message, level));\n  }\n\n  private _reportExpressionParserErrors(errors: ParserError[], sourceSpan: ParseSourceSpan) {\n    for (const error of errors) {\n      this._reportError(error.message, sourceSpan);\n    }\n  }\n\n  // Make sure all the used pipes are known in `this.pipesByName`\n  private _checkPipes(ast: ASTWithSource, sourceSpan: ParseSourceSpan): void {\n    if (ast && this.pipesByName) {\n      const collector = new PipeCollector();\n      ast.visit(collector);\n      collector.pipes.forEach((ast, pipeName) => {\n        const pipeMeta = this.pipesByName!.get(pipeName);\n        if (!pipeMeta) {\n          this._reportError(\n              `The pipe '${pipeName}' could not be found`,\n              new ParseSourceSpan(\n                  sourceSpan.start.moveBy(ast.span.start), sourceSpan.start.moveBy(ast.span.end)));\n        } else {\n          this._usedPipes.set(pipeName, pipeMeta);\n        }\n      });\n    }\n  }\n\n  /**\n   * @param propName the name of the property / attribute\n   * @param sourceSpan\n   * @param isAttr true when binding to an attribute\n   */\n  private _validatePropertyOrAttributeName(\n      propName: string, sourceSpan: ParseSourceSpan, isAttr: boolean): void {\n    const report = isAttr ? this._schemaRegistry.validateAttribute(propName) :\n                            this._schemaRegistry.validateProperty(propName);\n    if (report.error) {\n      this._reportError(report.msg!, sourceSpan, ParseErrorLevel.ERROR);\n    }\n  }\n}\n\nexport class PipeCollector extends RecursiveAstVisitor {\n  pipes = new Map<string, BindingPipe>();\n  override visitPipe(ast: BindingPipe, context: any): any {\n    this.pipes.set(ast.name, ast);\n    ast.exp.visit(this);\n    this.visitAll(ast.args, context);\n    return null;\n  }\n}\n\nfunction isAnimationLabel(name: string): boolean {\n  return name[0] == '@';\n}\n\nexport function calcPossibleSecurityContexts(\n    registry: ElementSchemaRegistry, selector: string, propName: string,\n    isAttribute: boolean): SecurityContext[] {\n  const ctxs: SecurityContext[] = [];\n  CssSelector.parse(selector).forEach((selector) => {\n    const elementNames = selector.element ? [selector.element] : registry.allKnownElementNames();\n    const notElementNames =\n        new Set(selector.notSelectors.filter(selector => selector.isElementSelector())\n                    .map((selector) => selector.element));\n    const possibleElementNames =\n        elementNames.filter(elementName => !notElementNames.has(elementName));\n\n    ctxs.push(...possibleElementNames.map(\n        elementName => registry.securityContext(elementName, propName, isAttribute)));\n  });\n  return ctxs.length === 0 ? [SecurityContext.NONE] : Array.from(new Set(ctxs)).sort();\n}\n\n/**\n * Compute a new ParseSourceSpan based off an original `sourceSpan` by using\n * absolute offsets from the specified `absoluteSpan`.\n *\n * @param sourceSpan original source span\n * @param absoluteSpan absolute source span to move to\n */\nfunction moveParseSourceSpan(\n    sourceSpan: ParseSourceSpan, absoluteSpan: AbsoluteSourceSpan): ParseSourceSpan {\n  // The difference of two absolute offsets provide the relative offset\n  const startDiff = absoluteSpan.start - sourceSpan.start.offset;\n  const endDiff = absoluteSpan.end - sourceSpan.end.offset;\n  return new ParseSourceSpan(\n      sourceSpan.start.moveBy(startDiff), sourceSpan.end.moveBy(endDiff),\n      sourceSpan.fullStart.moveBy(startDiff), sourceSpan.details);\n}\n"]} |
---|